问题描述:
JBoss JDBC驱动时,第一次建立如果库连接会抛出类似于下面的异常
21:21:36,666 WARN [JBossManagedConnectionPool] Throwable while attempting to get a new connection: null
org.jboss.resource.JBossResourceException: Could not create connection; - nested throwable: (org.jboss.resource.JBossResourceException: Apparently wrong driver class specified for URL: class: oracle.jdbc.driver.OracleDriver, url: jdbc:oracle:thin:@ 10.0.32 .25:1521:sid)
at org.jboss.resource.adapter.jdbc.local.LocalManagedConnectionFactory.createManagedConnection(LocalManagedConnectionFactory.java:179)
at org.jboss.resource.connectionmanager.InternalManagedConnectionPool.createConnectionEventListener(InternalManagedConnectionPool.java:565)
但在第二次取得数据库连接时却是正常的。
解决方法:
将JBoss JDBC驱动拷贝到jboss_server_home/lib目录下。这个方法对于每一个jboss环境都需要进行这样的拷贝,比较麻烦。
问题分析:
出现这个问题的JBoss服务器配置文件jboss_server_home/deploy/jbossweb-tomcat55.sar/META-INF/jboss-service.xml里的UseJBossWebLoader配置项应该都是配置成为false的,即使用的并不是JBoss共享扁平的ClasssLoader并且jboss_server_home/lib不包含JBoss JDBC驱动。下面我们来分析造成这个问题的具体原因。
首先我们来了解一下JBoss JDBC驱动的管理。JBoss JDBC驱动程序在载入的时候都会通过java.sql.DriverManager.registerDriver(Driver)方法将自身注册到驱动管理器中。在注册后我们就可以通过DriverManager.getDriver(String url)方法取得能够处理传入的数据库url的驱动程序,或者通过DriverManager.getConnection(String url, String user, String password)方法取得url对应驱动的连接。但在这里有一个问题需要我们注意,在取得连接或者驱动的时候,它需要从已注册的驱动里选择合适的驱动程序出来。这个合适的驱动总结出来有两条
一是当前调用getDriver或getConnect方法的类的ClassLoader能够载入相应的JBoss JDBC驱动程序,并且载入的JBoss JDBC驱动程序类要和已注册的驱动程序类相等,调用getDriver或getConnect方法的类ClassLoader通过本地方法DriverManager.getCallerClassLoader()获得,它得到调用类的ClassLoader。
二是对于符合前面条件的驱动,还需要判断当前这个驱动能不能处理连接url,如果不符合,则从注册的驱动里查找下一次驱动,如果满足,则使用这个驱动来创建连接。
【编辑推荐】