前几天同事说项目出问题了,请求一直报错,我看了下服务器日志,发现服务器一直报错Caused by: com.alibaba.druid.pool.GetConnectionTimeoutException: wait millis 60000, active 20, maxActive 20, creating 0
大致意思是druid的连接数(20)已经达到了最大连接数(20),无法提供数据库连接。我就很疑惑,目前我们项目的并发量也不是很高,怎么会出这种问题呢?而且这20个连接一直都不会释放,出现了这个错误之后,关于数据库的接口全部用不了了。
然后我重启了一下服务器,可以正常访问了。但是过了几天之后,还是出现了这个问题,我就去网上查了些解决方案,配置了druid的回收超时的连接设置:
properties配置:
#druid recycle Druid的连接回收机制
#不建议在生产环境中使用,会影响性能,仅用于连接泄露检测诊断
#超过时间限制是否回收
spring.datasource.druid.removeAbandoned = true
#超时时间;单位为秒。180秒=3分钟
spring.datasource.druid.removeAbandonedTimeout = 180
#关闭abanded连接时输出错误日志
spring.datasource.druid.logAbandoned = true
xml配置:
<!-- RemoveAbandanded功能不建议在生产环境中使用,会影响性能,仅用于连接泄露检测诊断 -->
<property name="removeAbandoned" value="true" /> <!-- 打开removeAbandoned功能 -->
<property name="removeAbandonedTimeout" value="180" /> <!-- 180秒,也就是3分钟 -->
<property name="logAbandoned" value="true" /> <!-- 打开abanded连接时输出错误日志 -->
结果第二天日志中果然出现了druid回收超时连接的错误:
检查了代码后得到结论如下:
1、WeatherService类上加了@Transactional注解,进入这个方法就开始事务,申请了数据库连接。
2、这个方法内调用了BaseUtils的httpGet方法,发起了http请求,某些原因(网络异常之类的)导致这个请求一直卡住(没有设置请求超时时间,所以它会一直卡住),这个线程就一直卡住在这里,所以数据库连接就一直被占用着不释放了。那么如果来20个这种请求,数据库连接池里面的连接就被占完了。。。。
解决方法:
设置申请连接超时时间和socket响应超时时间,如果超过时间会报错
HttpGet httpGet = new HttpGet(url);
// 设置请求超时时间
RequestConfig requestConfig = RequestConfig.custom().setConnectTimeout(5000) // 设置申请连接超时时间,单位毫秒。.setSocketTimeout(5000) //如果socket响应超时时间,单位毫秒.build();
httpGet.setConfig(requestConfig);
第二天一看日志,果然是报了请求超时的错,没有报连接池占满的错了。
···············至此问题解决················