dbcp mysql 8小时_使用c3p0与DBCP连接池,造成的MySql 8小时问题解决方式

本文提供了对c3p0与DBCP连接池连接MySql数据库时。 8小时内无请求自己主动断开连接的解决方式。首先介绍一下我在项目(c3p0连接池)中遇到的问题,后面还提供了使用DBCP连接池的解决方式。

基本问题解决

项目环境:

Java Web项目框架为Spring MVC+JPA,使用c3p0连接池,公布环境为Tomcat 7

错误描写叙述:

项目执行一段时间(大概几个小时)之后訪问时会出现第一次訪问报错,再次訪问正常的现象。且多次出现此问题。

报错日志:

org.springframework.transaction.CannotCreateTransactionException: Could not open JPA EntityManager for transaction; nested exception is javax.persistence.PersistenceException: org.hibernate.TransactionException: JDBC begin transaction failed:

at org.springframework.orm.jpa.JpaTransactionManager.doBegin(JpaTransactionManager.java:428)

at org.springframework.transaction.support.AbstractPlatformTransactionManager.getTransaction(AbstractPlatformTransactionManager.java:372)

at org.springframework.transaction.interceptor.TransactionAspectSupport.createTransactionIfNecessary(TransactionAspectSupport.java:417)

at org.springframework.transaction.interceptor.TransactionAspectSupport.invokeWithinTransaction(TransactionAspectSupport.java:255)

at org.springframework.transaction.interceptor.TransactionInterceptor.invoke(TransactionInterceptor.java:94)

at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:172)

at org.springframework.aop.framework.CglibAopProxy$DynamicAdvisedInterceptor.intercept(CglibAopProxy.java:631)

at com.appcarcare.cube.service.UserService$$EnhancerByCGLIB$$a4429cba.getUserDao()

at com.appcarcare.cube.servlet.DataCenterServlet$SqlTimer.connectSql(DataCenterServlet.java:76)

at com.appcarcare.cube.servlet.DataCenterServlet$SqlTimer.run(DataCenterServlet.java:70)

at java.util.TimerThread.mainLoop(Timer.java:555)

at java.util.TimerThread.run(Timer.java:505)

Caused by: javax.persistence.PersistenceException: org.hibernate.TransactionException: JDBC begin transaction failed:

at org.hibernate.ejb.AbstractEntityManagerImpl.convert(AbstractEntityManagerImpl.java:1387)

at org.hibernate.ejb.AbstractEntityManagerImpl.convert(AbstractEntityManagerImpl.java:1310)

at org.hibernate.ejb.AbstractEntityManagerImpl.throwPersistenceException(AbstractEntityManagerImpl.java:1397)

at org.hibernate.ejb.TransactionImpl.begin(TransactionImpl.java:62)

at org.springframework.orm.jpa.DefaultJpaDialect.beginTransaction(DefaultJpaDialect.java:71)

at org.springframework.orm.jpa.vendor.HibernateJpaDialect.beginTransaction(HibernateJpaDialect.java:60)

at org.springframework.orm.jpa.JpaTransactionManager.doBegin(JpaTransactionManager.java:378)

... 11 more

Caused by: org.hibernate.TransactionException: JDBC begin transaction failed:

at org.hibernate.engine.transaction.internal.jdbc.JdbcTransaction.doBegin(JdbcTransaction.java:76)

at org.hibernate.engine.transaction.spi.AbstractTransactionImpl.begin(AbstractTransactionImpl.java:160)

at org.hibernate.internal.SessionImpl.beginTransaction(SessionImpl.java:1426)

at org.hibernate.ejb.TransactionImpl.begin(TransactionImpl.java:59)

... 14 more

Caused by: com.mysql.jdbc.exceptions.jdbc4.CommunicationsException: Communications link failure

The last packet successfully received from the server was 1,836,166 milliseconds ago. The last packet sent successfully to the server was 29,134 milliseconds ago.

at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)

at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:57)

at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45)

at java.lang.reflect.Constructor.newInstance(Constructor.java:526)

at com.mysql.jdbc.Util.handleNewInstance(Util.java:411)

at com.mysql.jdbc.SQLError.createCommunicationsException(SQLError.java:1117)

at com.mysql.jdbc.MysqlIO.reuseAndReadPacket(MysqlIO.java:3567)

at com.mysql.jdbc.MysqlIO.reuseAndReadPacket(MysqlIO.java:3456)

at com.mysql.jdbc.MysqlIO.checkErrorPacket(MysqlIO.java:3997)

at com.mysql.jdbc.MysqlIO.sendCommand(MysqlIO.java:2468)

at com.mysql.jdbc.MysqlIO.sqlQueryDirect(MysqlIO.java:2629)

at com.mysql.jdbc.ConnectionImpl.execSQL(ConnectionImpl.java:2713)

at com.mysql.jdbc.ConnectionImpl.setAutoCommit(ConnectionImpl.java:5060)

at com.mchange.v2.c3p0.impl.NewProxyConnection.setAutoCommit(NewProxyConnection.java:881)

at org.hibernate.engine.transaction.internal.jdbc.JdbcTransaction.doBegin(JdbcTransaction.java:72)

... 17 more

Caused by: java.net.SocketException: Software caused connection abort: recv failed

at java.net.SocketInputStream.socketRead0(Native Method)

at java.net.SocketInputStream.read(SocketInputStream.java:150)

at java.net.SocketInputStream.read(SocketInputStream.java:121)

at com.mysql.jdbc.util.ReadAheadInputStream.fill(ReadAheadInputStream.java:114)

at com.mysql.jdbc.util.ReadAheadInputStream.readFromUnderlyingStreamIfNecessary(ReadAheadInputStream.java:161)

at com.mysql.jdbc.util.ReadAheadInputStream.read(ReadAheadInputStream.java:189)

at com.mysql.jdbc.MysqlIO.readFully(MysqlIO.java:3014)

at com.mysql.jdbc.MysqlIO.reuseAndReadPacket(MysqlIO.java:3467)

... 25 more

原因分析:

MySQLserver默认的“wait_timeout”是28800秒即8小时,意味着假设一个连接的空暇时间超过8个小时。MySQL将自己主动断开该连接,而连接池却觉得该连接还是有效的(由于并未校验连接的有效性),当应用申请使用该连接时,就会导致上面的报错。

解决方式(解决问题的办法有三种,推荐另外一种):

1. 添加 MySQL 的 wait_timeout 属性的值

改动mysql安装文件夹下的配置文件 my.ini文件(假设没有此文件,复制“my-default.ini”文件,生成“复件 my-default.ini”文件。将“复件

my-default.ini”文件重命名成“my.ini”),在文件里设置:

wait_timeout=31536000

interactive_timeout=31536000

这两个參数的默认值是8小时(60*60*8=28800)。

注意:1.wait_timeout的最大值仅仅同意2147483 (24天左右)

2.改动配置文件为网上大部分文章所提供的方式,也能够使用mysql命令对这两个属性进行改动

af1c08ed01d47852cdfb88c35ee70da5.png

6877080.html

6877080.html

2. 降低连接池内连接的生存周期

降低连接池内连接的生存周期,使之小于上一项中所设置的wait_timeout

的值。

改动 c3p0 的配置文件。在 Spring 的配置文件里设置:

3. 定期使用连接池内的连接

定期使用连接池内的连接,使得它们不会由于闲置超时而被 MySQL 断开。

改动 c3p0 的配置文件,在Spring 的配置文件里设置:

知识扩展

C3P0

C3P0是一个开放源码的JDBC连接池,它在lib文件夹中与Hibernate一起公布,包含了实现jdbc3和jdbc2扩展规范说明的Connection 和Statement 池的DataSources 对象。c3p0配置文件

3

30

1000

false

Test

false

100

null

false

60

3

60

15

100

3

root

password

select id from test where id=1

300

false

true

root

在Hibernate(spring管理)中的配置:

oracle.jdbc.driver.OracleDriver

jdbc:oracle:thin:@localhost:1521:Test

Kay

root

###########################

### C3P0 Connection Pool###

###########################

#hibernate.c3p0.max_size 2

#hibernate.c3p0.min_size 2

#hibernate.c3p0.timeout 5000

#hibernate.c3p0.max_statements 100

#hibernate.c3p0.idle_test_period 3000

#hibernate.c3p0.acquire_increment 2

#hibernate.c3p0.validate false

在hibernate.cfg.xml文件中面增加例如以下的配置:

20

5

120

100

120

2

true

使用DBCP连接池时出现MySql 8小时断开连接的解决方法

改动l配置文件:

改动例如以下:

 当中testOnBorrow和 validationQuery 非常重要。

testOnBorrow的意思是从数据库连接池中取得连接时,对其的有效性进行检查。

validationQuery 是用来检查的SQL语句,“select 1”运行较快。是一个不错的检測语句。

回想

当測试人员反映到这个问题的时候。非常快就锁定了Mysql八小时的问题。可是解决方式却费了我不小功夫。先是考虑到改动mysql配置文件肯定是不太合理的。弃之。

然后想了一下在一个servlet中写了个定时器,让它每两小时查一下数据库,执行了几天发现问题仍然存在,将定时器间隔时间改动为30分钟、3分钟仍然无济于事,异常照常出现,弃之。

在网上搜了一下解决方式也挺多。把

写入到配置文件里,执行报错。说bean中没有这两个属性,查看了一下原来是我用的c3p0连接池。而这个解决方式是针对DBCP连接池的。

最后在配置文件里。加入了

成功攻克了问题。

感悟,问题非常easy。就是改个配置文件。可是遇到问题须要高速解决这个问题的能力,更须要认真的态度。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,这是我对你的问题的回答: 常用的数据库连接池DBCPc3p0 和 Druid。它们都是用来提高数据库连接的效率和性能的。 以下是它们的配置说明: 1. DBCP DBCP 是 Apache 开源组织提供的连接池,配置方式如下: ``` <!-- 引入 DBCP 依赖 --> <dependency> <groupId>commons-dbcp</groupId> <artifactId>commons-dbcp</artifactId> <version>1.4</version> </dependency> <!-- 配置 DBCP 连接池 --> <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource"> <property name="driverClassName" value="com.mysql.jdbc.Driver" /> <property name="url" value="jdbc:mysql://localhost:3306/test" /> <property name="username" value="root" /> <property name="password" value="root" /> <property name="maxActive" value="100" /> <property name="maxIdle" value="30" /> <property name="maxWait" value="10000" /> </bean> ``` 2. c3p0 c3p0 是一个开源的 JDBC 连接池,配置方式如下: ``` <!-- 引入 c3p0 依赖 --> <dependency> <groupId>c3p0</groupId> <artifactId>c3p0</artifactId> <version>0.9.5.2</version> </dependency> <!-- 配置 c3p0 连接池 --> <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"> <property name="driverClass" value="com.mysql.jdbc.Driver" /> <property name="jdbcUrl" value="jdbc:mysql://localhost:3306/test" /> <property name="user" value="root" /> <property name="password" value="root" /> <property name="maxPoolSize" value="100" /> <property name="minPoolSize" value="30" /> <property name="maxIdleTime" value="10000" /> </bean> ``` 3. Druid Druid 是阿里巴巴开源的 JDBC 连接池,配置方式如下: ``` <!-- 引入 Druid 依赖 --> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.1.10</version> </dependency> <!-- 配置 Druid 连接池 --> <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"> <property name="driverClassName" value="com.mysql.jdbc.Driver" /> <property name="url" value="jdbc:mysql://localhost:3306/test" /> <property name="username" value="root" /> <property name="password" value="root" /> <property name="initialSize" value="10" /> <property name="maxActive" value="100" /> <property name="minIdle" value="30" /> <property name="maxWait" value="10000" /> </bean> ``` 以上就是这些连接池的基本配置方式。具体的配置可以参考各个连接池的官方文档。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值