DBCP数据库链接失效的解决方法(Io 异常:Connection reset)

网上不少评论说DBCP有不少BUG,可是都没有指明是什么BUG,只有一部分人说数据库若是由于某种缘由断掉后再DBCP取道的链接都是失效的链接,而没有从新取。有的时候会报Io 异常:Connection reset。

解决方法: spring

spring中datasource的配置以下:
    <bean id="dispatchdataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close"> 
    <property name="driverClassName" value="oracle.jdbc.driver.OracleDriver" /> 
    <property name="url" value="jdbc:oracle:thin:@127.0.0.1 :1521:myserver" /> 
    <property name="username" value="user1" /> 
    <property name="password" value="pwd" /> 
    <property name="maxActive" value="10000" /> 
    <property name="maxIdle" value="30" /> 
     <property name="minIdle" value="2" /> 
    <property name="maxWait" value="600000" /> 
    <property name="testOnBorrow" value="true"/> 
    <property name="testWhileIdle" value="true"/> 
    <property name="validationQuery" value="select 1 from dual"/> 
</bean>
sql

分析: 数据库

DBCP使用apache的对象池ObjectPool做为链接池的实现,有如下主要的方法 apache

Object borrowObject() throws Exception;从对象池取得一个有效对象 oracle

void returnObject(Object obj) throws Exception;使用完的对象放回对象池 url

void invalidateObject(Object obj) throws Exception;使对象失效 spa

void addObject() throws Exception;生成一个新对象 .net


ObjectPool的一个实现就是GenericObjectPool,这个类使用对象工厂PoolableObjectFactory实现对象的生成,失效检查等等功能,以其实现数据库链接工厂PoolableConnectionFactory作以说明,主要方法: 线程

     Object makeObject() throws Exception; 使用ConnectionFactory生成新链接 server

     void destroyObject(Object obj) throws Exception;关闭链接

     boolean validateObject(Object obj); 验证链接是否有效,若是_validationQuery不空,则使用该属性做为验证链接是否有效的sql语句,查询数据库

     void activateObject(Object obj) throws Exception;激活链接对象

     void passivateObject(Object obj) throws Exception; 关闭链接生成过的Statement和ResultSet,使链接处于非活动状态

    而GenericObjectPool有几个主要属性

     _timeBetweenEvictionRunsMillis:失效检查线程运行时间间隔,默认-1

     _maxIdle:对象池中对象最大个数

     _minIdle:对象池中对象最小个数

     _maxActive:能够从对象池中取出的对象最大个数,为0则表示没有限制,默认为8

 

     在构造GenericObjectPool时,会生成一个内嵌类Evictor,实现自Runnable接口。若是 _timeBetweenEvictionRunsMillis大于0,每过_timeBetweenEvictionRunsMillis毫秒 Evictor会调用evict()方法,检查对象的闲置时间是否大于 _minEvictableIdleTimeMillis毫秒(_minEvictableIdleTimeMillis小于等于0时则忽略,默认为30 分钟),是则销毁此对象,不然就激活并校验对象,而后调用ensureMinIdle方法检查确保池中对象个数不小于_minIdle。在调用 returnObject方法把对象放回对象池,首先检查该对象是否有效,而后调用PoolableObjectFactory 的passivateObject方法使对象处于非活动状态。再检查对象池中对象个数是否小于_maxIdle,是则能够把此对象放回对象池,不然销毁此对象。

 

     还有几个很重要的属性,_testOnBorrow、_testOnReturn、_testWhileIdle,这些属性的意义是取得、返回对象和空闲时是否进行验证,检查对象是否有效,默认都为false即不验证。因此当使用DBCP时,数据库链接由于某种缘由断掉后,再从链接池中取得链接又不进行验证,这时取得的链接实际已经时无效的数据库链接了。网上不少说 DBCP的bug应该都是如此吧,只有把这些属性设为true,再提供_validationQuery语句就能够保证数据库链接始终有效了,oracle数据库能够使用SELECT COUNT(*) FROM DUAL,不过DBCP要求_validationQuery语句查询的记录集必须不为空,可能这也能够算一个小小的BUG,其实只要_validationQuery语句执行经过就能够了。

相关文章
相关标签/搜索