`
agapple
  • 浏览: 1597866 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

解读dbcp自动重连那些事

阅读更多

可以后另一篇做对比:http://agapple.iteye.com/blog/772507


同样的内容,不同的描述方式,不一样的效果.

 

Hi all :

最近在做 offerdetail 优化时,替换了数据库驱动,从 c3p0 0.9.1 -> dbcp 1.4 顺便研究了下 dbcp 的自动重连的一套机制,也做一下分享,大家周知一下。

 

数据库链接 常见的问题:

1. 数据库意外重启后,原先的数据库连接池能自动废弃老的无用的链接,建立新的数据库链接

2. 网络异常中断后,原先的建立的 tcp 链接,应该能进行自动切换。比如网站演习中的交换机重启会导致网络瞬断

3. 分布式数据库中间件,比如 cobar 会定时的将空闲链接异常关闭,客户端会出现半开的空闲链接。

 

大致思考解决思路:

1.      sql 心跳检查 ( 主动式 )

2.      拿链接尝试一下,发现处理失败丢弃链接,探雷的请求会失败几个  ( 牺牲小我,完成大我的精神 )

3.      设置合理的空闲链接的超时时间,避免半开链接 ( 懒模式,解决半开链接 )

 

 

下面我们来看看,在 dbcp 中是如何实现。

sql 心跳检查

sql validate 配置

<property name= "testWhileIdle" ><value> true </value></property>

<property name= "testOnBorrow" ><value> false </value></property>

<property name= "testOnReturn" ><value> false </value></property>

<property name= "validationQuery" ><value>select sysdate from dual</value></property>

<property name= "validationQueryTimeout" ><value>1</value></property>

<property name= "timeBetweenEvictionRunsMillis" ><value>30000</value></property>

<property name= "numTestsPerEvictionRun" ><value>16</value></property>

参数说明

  

   dbcp 是采用了 commons-pool 做为其连接池管理, testOnBorrow,testOnReturn, testWhileIdle pool 是提供的几种校验机制,通过外部钩子的方式回调 dbcp 的相关数据库链接 (validationQuery) 校验 , dbcp 相关外部钩子类: PoolableConnectionFactory, 继承于 common-pool PoolableObjectFactory , dbcp 通过 GenericObjectPool 这一入口,进行连接池的 borrow,return 处理。

具体参数描述:

   1. testOnBorrow : 顾明思义,就是在进行borrowObject进行处理时,对拿到的connection进行validateObject校验

   2. testOnReturn : 顾明思义,就是在进行returnObject对返回的connection进行validateObject校验,个人觉得对数据库连接池的管理意义不大

   3. testWhileIdle : 关注的重点,GenericObjectPool中针对pool管理,起了一个 异步Evict的TimerTask定时线程进行控制 ( 可通过设置参数 timeBetweenEvictionRunsMillis>0), 定时对线程池中的链接进行validateObject校验,对无效的链接进行关闭后,会调用ensureMinIdle,适当建立链接保证最小的minIdle连接数。

   4. timeBetweenEvictionRunsMillis, 设置的Evict线程的时间,单位ms,大于0才会开启evict检查线程

   5. validateQuery , 代表检查的sql

   6. validateQueryTimeout , 代表在执行检查时,通过statement设置,statement.setQueryTimeout(validationQueryTimeout)

   7. numTestsPerEvictionRun ,代表每次检查链接的数量,建议设置和maxActive一样大,这样每次可以有效检查所有的链接.

Sql 心跳检查几点思考:

1. 性能问题。

目前网站的应用大部分的瓶颈还是在I/O这一块,大部分的I/O还是在数据库的这一层面上,每一个请求可能会调用10来次SQL查询,如果不走事务,一个请求会重复获取链接,如果每次获取链接,比如在testOnBorrow都进行validateObject,性能开销不是很能接受,可以假定一次SQL操作消毫0.5~1ms(一般走了网络请求基本就这数)

2 .成本和收益

网站异常数据库重启,网络异常断开的频率是非常低的,一般也就在数据库升级,演习维护时才会进行,而且一般也是选在晚上,访问量相对比较低的请求,而且一般会有人员值班关注,所以异步的validateObject是可以接受,但一个前提需要确保能保证在一个合理的时间段内,数据库能完成自动重联。

 

请求探雷

相关配置

dbcp 自身默认支持,不需要配置

原理描述

common-pools 通过borrowObject , returnObject完成连接的获取和释放,正常的情况是一次请求中borrow和return是一对的,有借就有还。

但在准备returnObject时,dbcp会做一件事,就是看看这个object是否已经是坏了的,如果坏了就直接丢了,就直接给丢弃了。

 

代码层面:

1. 在dbcp中PoolingDataSource(实现DataSource接口)调用 PoolableConnection(dbcp connnection 相关的pool delegate操作)进行相应关闭时,会检查 _conn.isClosed() ,针对DataSource如果isClosed返回为 true的则不调用returnObject,直接丢弃了链接。

2. _conn.isClosed()是否保险,从jdk的api描述中: A connection is closed if the method close has been called on it or if certain fatal errors have occurred. 里面提供两种情况,一种就是被调用了closed方法,另一种就是出现一些异常,说的比较含糊。

 

空闲链接检查

相关配置

<property name="minEvictableIdleTimeMillis "><value>18000000</value></property>

<property name="removeAbandoned" ><value>true</value></property> 

<property name="removeAbandonedTimeout "><value>180</value></property>

参数说明

1. minEvictableIdleTimeMillis  dbcp默认是30分,需要开启异步线程Evict,否则不生效。原理很简单,就是通过一个异步线程,每次检查connnection上一次使用的时间戳,看看是否已经超过这个timeout时间设置。

2. removeAbandoned , removeAbandonedTimeout ,主要是用于在出现链接紧张时候,会扫描一些链接未超过removeAbandonedTimeout时间还未被释放,会主动的关闭该链接。

适用情况

1. 我们使用的cobar后端会有定时关闭空闲链接的操作,默认的空闲链接timeout时间为1小时,和其他oracle , mysql 各不相同,所以设置好这个空闲链接的timeout时间还是挺重要.

 

2. 一般会是几种情况出现需要removeAbandoned: 

* 代码未在finally释放connection ,  不过我们都用sqlmapClientTemplate,底层都有链接释放的过程

* 遇到数据库死锁 。以前遇到过后端存储过程做了锁表操作,导致前台集群中连接池全都被block住,后续的业务处理因为拿不到链接所有都处理失败了。

 

 

聊聊 c3p0 配置

还有我们配置的c3p0所谓的自动重连的3个参数,

<prop key="acquireRetryAttempts">30</prop>

    <prop key="acquireRetryDelay">1000</prop>

    <prop key="breakAfterAcquireFailure">false</prop>

 

个人觉得就是一个误导 ,这几个配置只是在从连接池获取链接时,获取失败多尝试几次,因为我们从pool从获取链接最多只会等待固定timeout时间。

如果要达到自动重连的效果,必须要c3p0支持请求探雷或者是sql心跳检查功能,能自动的剔除无效的链接。 

可见c3p0官方文档描述:http://www.mchange.com/projects/c3p0/index.html#configuring_recovery

 

最后:

Dbcp 将是我们以后数据库驱动选择的趋势,最后我们如何选择如何自动重连,这个也得根据我们的应用场景而定。比如只读的web系统,后台业务系统,任务系统可能处理方式就不同。

只读Web系统:可采取请求探雷的策略,也就失败连接池个数的请求,失败了页面刷新一次就好。

后台业务系统:一般业务都涉及数据库的写操作,很多数据不可重入,一次处理失败后就只能靠手工干预处理。这时候得考虑是否需要使用sql心跳检查,比如testOnBorrow或者testWhileIdle.

分享到:
评论
10 楼 agapple 2012-05-31  
wangyibao 写道
还有,不知道楼主对于jboss的ExceptionSorter有没有研究过,我也很想知道它与DBCP的这种有效检测优劣比较。为什么DBCP没有这种ExcetpionSorter的实现?


这个我倒没看过jboss的实现,单从技术实现成本看并不复杂。而且早期的jboss的连接池实现就是在dbcp的基础上包上了一层壳。 

dbcp默认会捕获底层socket异常,判断当前的tcp链接isClosed后,自动关闭链接。
9 楼 wangyibao 2012-05-31  
还有,不知道楼主对于jboss的ExceptionSorter有没有研究过,我也很想知道它与DBCP的这种有效检测优劣比较。为什么DBCP没有这种ExcetpionSorter的实现?
8 楼 wangyibao 2012-05-31  
楼主,有个问题困扰着我,我看源代码也没有发现。
如果没有配置DBCP的有效检测,那么数据库重启,APP持有了已经被closed掉的连接,那么一旦APP用这些连接时,发现抛SQLException,我想知道,此时Pool会不会drop掉这些被closed的连接Obect?如果drop,那么DBCP怎么实现的?如果不drop,从表面现象看,又好像能够重新连接正常。
7 楼 C_J 2010-12-01  
    看过Common Pool的源码,要是有时间,也想看看DBCP的源码,Common Pool基本上是管理所有Object的容器,提供几种常用的容器,比如Stack,List,Key-Value等,而对外主要扩展的地方是PoolFactory,以满足用户根据不同的策略和要求对对象进行borrow和return了:)

    另外,感谢shared!
6 楼 agapple 2010-11-02  
rustlingwind 写道
agapple 写道
正因为链接是后进先出的,所以你的现象如果真是第一次请求,确认当时没有其他线程,我还真想不到很好的解释,可以查看下当时的网络tcp状态。
有问题欢迎继续交流,大家一起研究下dbcp的一些代码。


楼主你好:
    非常感谢你的耐心解答。我再看一下是否是有其他线程将异常数据库连接占用。这一点我还真没注意。
    另外,我发现 commons-dbcp 和 commons-pool 的源码中有大量的锁,感觉性能应该受很大影响。比如jdk5的并发机制几乎没有体现。楼主是否知道有性能更好并且可靠性也不错的连接池啊?


恩,以前dbcp1.2性能更差,但在dbcp 1.4后已经有明显的改善,而且现在dbcp是由apache组织进行维护,现在的c3p0基本停滞不前,所以我们公司会准备将所有的c3p0切换到dbcp上。个人建议选择开源软件时需要考虑其稳定性,活跃性,性能需要综合考虑

其他的连接池有人做了比较,你可以参考下:http://www.iteye.com/topic/791358。
5 楼 rustlingwind 2010-11-01  
agapple 写道
正因为链接是后进先出的,所以你的现象如果真是第一次请求,确认当时没有其他线程,我还真想不到很好的解释,可以查看下当时的网络tcp状态。
有问题欢迎继续交流,大家一起研究下dbcp的一些代码。


楼主你好:
    非常感谢你的耐心解答。我再看一下是否是有其他线程将异常数据库连接占用。这一点我还真没注意。
    另外,我发现 commons-dbcp 和 commons-pool 的源码中有大量的锁,感觉性能应该受很大影响。比如jdk5的并发机制几乎没有体现。楼主是否知道有性能更好并且可靠性也不错的连接池啊?
4 楼 agapple 2010-10-26  
正因为链接是后进先出的,所以你的现象如果真是第一次请求,确认当时没有其他线程,我还真想不到很好的解释,可以查看下当时的网络tcp状态。
有问题欢迎继续交流,大家一起研究下dbcp的一些代码。
3 楼 agapple 2010-10-26  
rustlingwind 写道
楼主你好!我正在做数据库连接池的自动重连这块儿,看到你的文章,非常受用,但是也遇到一个问题。

我按照楼主的“testWhileIdle”配置:

<!-- sql 心跳 -->
<property name= "testWhileIdle" value="true"/>
<property name= "testOnBorrow" value="false"/>
<property name= "testOnReturn" value="false"/>
<property name= "validationQuery" value="select 1"/>
<property name= "validationQueryTimeout" value="1"/>
<property name= "timeBetweenEvictionRunsMillis" value="60000"/>
<property name= "numTestsPerEvictionRun" value="${jdbc.maxActive}"/>

试了一下。发现自动重连的确挺好用,但是也是非常奇怪,为何数据库重启后,没等到扫描线程去恢复连接池,竟然新的请求直接可以访问到数据库了。

我用的数据库是 mysql,跟踪日志发现,这个时候的确是重建了一个连接,不过对于数据库重启后的首次用户访问,也只是针对这次访问新建了一个连接而已。等到扫描线程的时间间隔到了,才恢复了整个连接池。

我很不明白,当拿到的连接是无效的时候,又是怎么当场重建的。看到你的解释:“正因为在获取异常链接后,因为做了_conn.isClosed()判断,所以异常链接并没有返回到连接池中,所以到数据库重启恢复后,每次都是调用pool重新构造一个新的connection,所以后面就正常了”。但是看完这段解释,在 commons-dbcp 和 commons-pool 的源代码中也没有找到具体的实现。

如果楼主有时间,能否详细解释一下这个地方,最好能够将源码中具体是在哪里实现的说一下,非常感谢!


hi , 看了下你的描述,是否存在这样的情况:数据库重启后的第一个链接,当时是否还有其他线程占用了原先的异常数据库链接,导致你请求的“第一个”链接重新去新建了一个请求?

几点说明:
1. dbcp默认对线程池的优先策略是LILO后进先出,它会将上一次使用后归还的链接,pool池的前前面一个,下次取就会拿到这一个,具体参数可以看GenericObjectPool._lifo参数,默认是后进先出,同样可选择为先进先出。
2. 你提的_conn.isClosed()判断,是在一次请求失败后,一般我们系统都采用了事务管理模板,会在最后finally释放connection,在dbcp中有一段释放代码:
public class PoolableConnection extends DelegatingConnection {
        public synchronized void close() throws SQLException {
        ......
        boolean isUnderlyingConectionClosed;
        try {
            isUnderlyingConectionClosed = _conn.isClosed();
        } catch (SQLException e) {
            try {
                _pool.invalidateObject(this); // XXX should be guarded to happen at most once
            } catch(IllegalStateException ise) {
                // pool is closed, so close the connection
                passivate();
                getInnermostDelegate().close();
            } catch (Exception ie) {
                // DO NOTHING the original exception will be rethrown
            }
            throw (SQLException) new SQLException("Cannot close connection (isClosed check failed)").initCause(e);
        }

        if (!isUnderlyingConectionClosed) {
            // Normal close: underlying connection is still open, so we
            // simply need to return this proxy to the pool
            try {
                _pool.returnObject(this); // XXX should be guarded to happen at most once
            } catch(IllegalStateException e) {
                // pool is closed, so close the connection
                passivate();
                getInnermostDelegate().close();
            } catch(SQLException e) {
                throw e;
            } catch(RuntimeException e) {
                throw e;
            } catch(Exception e) {
                throw (SQLException) new SQLException("Cannot close connection (return to pool failed)").initCause(e);
            }
        } else {
            // Abnormal close: underlying connection closed unexpectedly, so we
            // must destroy this proxy
            try {
                _pool.invalidateObject(this); // XXX should be guarded to happen at most once
            } catch(IllegalStateException e) {
                // pool is closed, so close the connection
                passivate();
                getInnermostDelegate().close();
            } catch (Exception ie) {
                // DO NOTHING, "Already closed" exception thrown below
            }
            throw new SQLException("Already closed.");
        }
    }
}

2 楼 rustlingwind 2010-10-25  
楼主你好!我正在做数据库连接池的自动重连这块儿,看到你的文章,非常受用,但是也遇到一个问题。

我按照楼主的“testWhileIdle”配置:

<!-- sql 心跳 -->
<property name= "testWhileIdle" value="true"/>
<property name= "testOnBorrow" value="false"/>
<property name= "testOnReturn" value="false"/>
<property name= "validationQuery" value="select 1"/>
<property name= "validationQueryTimeout" value="1"/>
<property name= "timeBetweenEvictionRunsMillis" value="60000"/>
<property name= "numTestsPerEvictionRun" value="${jdbc.maxActive}"/>

试了一下。发现自动重连的确挺好用,但是也是非常奇怪,为何数据库重启后,没等到扫描线程去恢复连接池,竟然新的请求直接可以访问到数据库了。

我用的数据库是 mysql,跟踪日志发现,这个时候的确是重建了一个连接,不过对于数据库重启后的首次用户访问,也只是针对这次访问新建了一个连接而已。等到扫描线程的时间间隔到了,才恢复了整个连接池。

我很不明白,当拿到的连接是无效的时候,又是怎么当场重建的。看到你的解释:“正因为在获取异常链接后,因为做了_conn.isClosed()判断,所以异常链接并没有返回到连接池中,所以到数据库重启恢复后,每次都是调用pool重新构造一个新的connection,所以后面就正常了”。但是看完这段解释,在 commons-dbcp 和 commons-pool 的源代码中也没有找到具体的实现。

如果楼主有时间,能否详细解释一下这个地方,最好能够将源码中具体是在哪里实现的说一下,非常感谢!
1 楼 rustlingwind 2010-10-25  
非常有用,十分感谢!;)楼主辛苦啦!

相关推荐

    jdbc与dbcp数据库连接

    注意,我们使用了try-with-resources语句,确保在处理完结果后自动关闭连接。 总的来说,JDBC是连接数据库的基础,而DBCP则是优化数据库访问的一种手段。结合使用JDBC和DBCP,开发者可以在Java应用中实现高效且易于...

    创建dbcp连接,dbcp(Spring)

    DBCP(Database Connection Pool)是Apache Commons项目中的一个数据库连接池组件,全称为Apache Commons DBCP。它允许开发者在应用程序中实现高效的数据库连接管理,通过复用已存在的数据库连接来减少每次请求时...

    dbcp连接jar包.zip

    DBCP(Database Connection Pool)是Apache的一个开源项目,名为Commons DBCP,它提供了一个数据库连接池的实现。在Java应用中,DBCP被广泛用于管理数据库连接,以提高性能和资源利用率。DBCP通过预先创建并维护一定...

    dbcp池连包(dbcp.jar)

    标题中的“dbcp池连包(dbcp.jar)”指的是Apache Commons DBCP,这是一个在Java环境中常用的数据库连接池组件。数据库连接池在应用服务器启动时创建一定数量的数据库连接放入池中,应用程序需要时从池中获取,使用...

    commons-dbcp-1.4

    Apache Commons DBCP 1.4 是一个在Java应用程序中管理数据库连接...尽管现在有更新的版本和替代品,如 HikariCP、C3P0 等,但 DBCP 仍然是许多项目中的首选,尤其是对于那些对性能要求不是特别高,但需要稳定性的应用。

    DBCP连接数据库的jar包

    DBCP(Database Connection Pool)是Apache Commons项目中的一个数据库连接池组件,全称为"Commons DBCP"。它提供了一种在Java应用程序中管理数据库连接的方法,通过池化技术来提高性能和效率。数据库连接池是现代...

    开源数据库连接池dbcp

    尽管DBCP在许多项目中得到了广泛应用,但它也有一些局限性,比如缺乏对现代数据库特性(如连接池自动缩放)的支持,以及相比其他更现代的连接池实现(如HikariCP、C3P0),性能可能稍逊一筹。因此,在选择数据库连接...

    dbcp所需要jar

    5. **使用DataSource**:在SpringMVC的控制器或服务层,可以通过@Autowired注解自动注入这个`dataSource`,然后通过JdbcTemplate或者NamedParameterJdbcTemplate进行数据库操作。例如: ```java @Autowired ...

    JDBC与DBCP连接mysql工程

    "JDBC与DBCP连接mysql工程"是一个专注于使用Java JDBC和DBCP(Apache Commons DBCP)连接MySQL数据库的学习资源。这里我们将详细探讨这两个关键技术及其在实际工程中的应用。 JDBC(Java Database Connectivity)是...

    DBCP依赖Jar包

    DBCP(Database Connection Pool)是Apache软件基金会的一个开源项目,全称为"Commons DBCP",它提供了一个数据库连接池服务。数据库连接池在应用服务器启动时建立一定数量的数据库连接,然后在应用程序需要时分配给...

    DBCP 数据库连接池JNDI连接 学习笔记

    数据库连接池(Database Connection Pool,简称 DBCP)是一种在多用户并发访问数据库时提高数据库系统性能的技术。它通过预创建并管理一定数量的数据库连接,避免了每次连接数据库时的初始化开销,同时也能有效地...

    java dbcp连接池

    Java DBCP(Database Connection Pool)是Apache软件基金会下的Jakarta项目提供的一种数据库连接池实现。它是基于JDBC(Java Database Connectivity)的,主要用于管理数据库连接,提高数据库操作的性能和效率。...

    commons-dbcp2-2.9.0-bin.zip

    DBCP的工作原理是,程序首先会初始化相应的数据库连接池,以供程序访问,当某个操作需要访问数据库时,程序会首先在连接池中取得空闲连接,如没有空闲连接在创建,用完之后归还连接池,这样达到了连接的重利用,不用...

    commons-dbcp-1.3

    3. 资源回收:当连接不再使用或者超过预设的空闲时间,DBCP会自动回收这些连接,释放资源。 4. 连接配置:开发者可以根据需求调整连接池的参数,如最大连接数、最小连接数、超时时间等,以适应不同的系统环境和负载...

    dbcp连接数据库

    DBCP(Database Connection Pool)是Apache软件基金会的Commons项目中的一个组件,它提供了一种数据库连接池的实现。数据库连接池在初始化时会创建一定数量的数据库连接,并将其保存在一个池中,当应用程序需要连接...

    commons-dbcp-1.4.zip

    如果一个连接长时间未被使用,DBCP可以将其标记为废弃并自动关闭,以防止资源泄露。 使用DBCP的好处主要包括: - **性能提升**: 通过复用已存在的连接,减少了数据库连接创建和销毁的开销。 - **资源优化**: 连接...

    dbcp jar包 dbcp jar 包

    DBCP(Database Connection Pool)是Apache软件基金会的一个开源项目,全称为Apache Commons DBCP,它提供了一个数据库连接池的实现。数据库连接池在多线程、高并发的环境中非常关键,因为它可以有效地管理和复用...

    DBCP数据库连接池jar包

    DBCP,全称为Apache Database Connection Pool,是由Apache软件基金会开发的一款开源数据库连接池组件。它在Java应用程序中扮演着至关重要的角色,通过有效地管理和复用数据库连接,显著提高了数据库访问性能,同时...

    连接池dbcp

    C3P0提供了更灵活的配置和自动管理,而HikariCP以其高速度和低资源占用著称。 8. **注意事项** 使用DBCP时,要确保正确处理异常,避免连接泄露。在程序结束时,最好关闭数据源,以释放所有资源。 总的来说,...

    commons-dbcp包资源

    4. 自动关闭:当不再需要连接时,DBCP会自动将连接返回到池中,而不是真正关闭它,以便后续请求可以再次使用。 5. 错误处理:如果连接出现异常,DBCP可以检测并移除损坏的连接,确保连接池的健康状态。 6. 资源...

Global site tag (gtag.js) - Google Analytics