记录一次线上事故:GetConnectionTimeoutException: wait millis 60000, active 20, maxActive 20, creating 0

        前几天同事说项目出问题了,请求一直报错,我看了下服务器日志,发现服务器一直报错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);

        第二天一看日志,果然是报了请求超时的错,没有报连接池占满的错了。

···············至此问题解决················

  • 3
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值