linux进入mysql查询彪哥,线上问题分析系列:数据库链接池内存泄漏问题的分析和解决方案...

前言

本文来自好朋友彪哥整理,实际的生产问题分析,绝对干货~java

1、问题描述

上周五晚上主营出现部分设备掉线,通过查看日志发现是因为缓存系统出现长时间gc致使的。这里的gc日志的特色是:mysql

1.gc时间都在2s以上,部分节点甚至出现12s超长时间gc。linux

2.同一个节点距离上次gc时间间隔为广泛为13~15天。面试

a98328b87f4c48d3b44670f231eaa59a.gif而后紧急把剩余未gc的一个节点内存dump下来,使用mat工具打开发现,com.mysql.jdbc.NonRegisteringDriver 对象占了堆内存的大部分空间。算法

a98328b87f4c48d3b44670f231eaa59a.gif查看对象数量,发现com.mysql.jdbc.NonRegisteringDriver$ConnectionPhantomReference 这个对象堆积了10140 个。

a98328b87f4c48d3b44670f231eaa59a.gifsql

初步判断长时间gc的问题应该是因为 com.mysql.jdbc.NonRegisteringDriver$ConnectionPhantomReference 这个对象大量堆积引发的。shell

2、问题分析

目前正式环境使用数据库相关依赖以下:数据库

依赖

版本

mysql

5.1.47

hikari

2.7.9

Sharding-jdbc

3.1.0

根据以上描述,提出如下问题:后端

一、com.mysql.jdbc.NonRegisteringDriver$ConnectionPhantomReference 究竟是个什么对象呢?缓存

二、这种对象为何会大量堆积,JVM回收不过来了?

NonRegisteringDriver$ConnectionPhantomReference 究竟是个什么对象呢?

简单来讲,NonRegisteringDriver类有个虚引用集合connectionPhantomRefs用于存储全部的数据库链接,NonRegisteringDriver.trackConnection方法负责把新建立的链接放入connectionPhantomRefs集合。源码以下:

1.public class NonRegisteringDriver implements java.sql.Driver {

2. protected static final ConcurrentHashMap connectionPhantomRefs = new ConcurrentHashMap();

3. protected static final ReferenceQueue refQueue = new ReferenceQueue();

4.

5. ....

6.

7. protected static void trackConnection(Connection newConn) {

8.

9. ConnectionPhantomReference phantomRef = new ConnectionPhantomReference((ConnectionImpl) newConn, refQueue);

10. connectionPhantomRefs.put(phantomRef, phantomRef);

11. }

12. ....

13. }

咱们追踪建立数据库链接的过程源码,发现其中会调到com.mysql.jdbc.ConnectionImpl的构造函数,该方法会调用createNewIO方法建立一个新的数据库链接MysqlIO对象,而后调用咱们上面提到的NonRegisteringDriver.trackConnection方法,把该对象放入NonRegisteringDriver.connectionPhantomRefs集合。源码以下:

1.public class ConnectionImpl extends ConnectionPropertiesImpl implements MySQLConnection {

2.

3. public ConnectionImpl(String hostToConnectTo, int portToConnectTo, Properties info, String databaseToConnectTo, String url) throws SQLException {

4. ...

5. createNewIO(false);

6. ...

7. NonRegisteringDriver.trackConnection(this);

8. ...

9. }

10.}

connectionPhantomRefs 是一个虚引用集合,何为虚引用?为何设计为虚引用队列

虚引用队列也称为“幽灵引用”,它是最弱的一种引用关系。

若是一个对象仅持有虚引用,那么它就和没有任何引用同样,在任什么时候候均可能被垃 圾回收器回收。

为一个对象设置虚 引用关联的惟一目的只是为了能在这个对象被收集器回收时收到一个系统通知。

当垃圾回收器准备回收一个对象时,若是发现它还有虚引用,就会在垃圾回收后,将这个虚引用加入引用队列,在其关联的虚引用出队前,不会完全销毁该对象。因此能够经过检查引用队列中是否有相应的虚引用来判断对象是否已经被回收了。

connectionPhantomRefs 这种对象为何会大量堆积,JVM回收不过来了?

这里结合项目中hikaricp数据配置和官方文档结合说明~

咱们先查阅hikaricp数据池的官网地址,看看部分属性介绍以下:

maximumPoolSize

This property controls the maximum size that the pool is allowed to reach, including both idle and in-use connections. Basically this value will determine the maximum number of actual connections to the database backend. A reasonable value for this is best determined by your execution environment. When the pool reaches this size, and no idle connections are available, calls to getConnection() will block for up to connectionTimeout milliseconds before timing out. Please read about pool sizing. Default: 10

maximumPoolSize控制最大链接数,默认为10

minimumIdle

This property controls the minimum number of idle connections that HikariCP tries to maintain in the pool. If the idle connections dip below this value and total connections in the pool are less than maximumPoolSize, HikariCP will make a best effort to add additional connections quickly and efficiently. However, for maximum performance and responsiveness to spike demands, we recommend not setting this value and instead allowing HikariCP to act as a fixed size connection pool. Default: same as maximumPoolSize

minimumIdle控制最小链接数,默认等同于maximumPoolSize,10。

⌚idleTimeout

This property controls the maximum amount of time that a connection is allowed to sit idle in the pool. This setting only applies when minimumIdle is defined to be less than maximumPoolSize. Idle connections will not be retired once the pool reaches minimumIdle connections. Whether a connection is retired as idle or not is subject to a maximum variation of +30 seconds, and average variation of +15 seconds. A connection will never be retired as idle before this timeout. A value of 0 means that idle connections are never removed from the pool. The minimum allowed value is 10000ms (10 seconds). Default: 600000 (10 minutes)

链接空闲时间超过idleTimeout(默认10分钟)后,链接会被抛弃

⌚maxLifetime

This property controls the maximum lifetime of a connection in the pool. An in-use connection will never be retired, only when it is closed will it then be removed. On a connection-by-connection basis, minor negative attenuation is applied to avoid mass-extinction in the pool. We strongly recommend setting this value, and it should be several seconds shorter than any database or infrastructure imposed connection time limit. A value of 0 indicates no maximum lifetime (infinite lifetime), subject of course to the idleTimeout setting. Default: 1800000 (30 minutes)

链接生存时间超过 maxLifetime(默认30分钟)后,链接会被抛弃.

咱们再回头看看项目的hikari配置:

配置了minimumIdle = 10,maximumPoolSize = 50,没有配置idleTimeout和maxLifetime。因此这两项会使用默认值 idleTimeout = 10分钟,maxLifetime = 30分钟。

也就是说假如数据库链接池已满,有50个链接,假如系统空闲,40个链接会在10分钟后(超过idleTimeout)被废弃;假如系统一直繁忙,50个链接会在30分钟后(超过maxLifetime)后被废弃。

猜想问题产生的根源:

每次新建一个数据库链接,都会把该链接放入connectionPhantomRefs集合中。数据链接在空闲时间超过idleTimeout或生存时间超过maxLifetime后会被废弃,在connectionPhantomRefs集合中等待回收。由于链接资源通常存活时间比较久,通过屡次Young GC,通常都能存活到老年代。若是这个数据库链接对象自己在老年代,connectionPhantomRefs中的元素就会一直堆积,直到下次 full gc。若是等到full gc 的时候connectionPhantomRefs集合的元素很是多,该次full gc就会很是耗时。

那么怎么解决呢?能够考虑优化minimumIdle、maximumPoolSize、idleTimeout、maxLifetime这些参数,下一小节咱们分析一波

3、问题验证

线上模拟环境

为了验证问题,咱们须要模拟线上环境,调整maxLifetime等参数~压测思路以下:

1.缓存系统模拟线上的配置,使用压测系统一段时间内持续压缓存系统,使缓存系统短期建立/废弃大量数据库链接,观察 NonRegisteringDriver 对象是否如期大量堆积,再手动调用 System.gc() 观察 NonRegisteringDriver 对象是否被清理。

2.调整maxLifetime 参数,观察相同的压测时间内 NonRegisteringDriver 对象是否还发生堆积。

这里有如下注意点:

一、 要知足 (gc 间隔时间 * 新生代进入老年代前的存活次数 < maxLifetime)这个条件,NonRegisteringDriver 对象才知足进入老年代的条件。

二、 minimumIdle = 10,maximumPoolSize = 50(minimumIdle和maximumPoolSize和线上配置一致),idleTimeout设置10s,maxLifetime设 100s(gc时间约20s,因此要大于 20 * 3 = 60s)。这样预计在持续压测下每30s就会产生10个新链接(就算设置了maximumPoolSize = 50,这种程序的压测10个链接足以应付)

三、 项目内存分配小一点,以及把新生代进入老年代前的存活次数调小一点,方便新生代的NonRegisteringDriver对象在较短期能进入老年代,方便在较短期观察到明显的对象增加。

四、 要监测缓存系统数据链接池的链接存活状况,以及系统 gc状况。

最终环境配置以下:

a98328b87f4c48d3b44670f231eaa59a.gif

模拟实验结果

启用jvisualvm工具对缓存系统进行实时观察

打开hikari相关debug日志观察链接池状况

设置 maxLifetime = 100s,启动缓存系统

确认hikari和jvm配置生效

a98328b87f4c48d3b44670f231eaa59a.gif

a98328b87f4c48d3b44670f231eaa59a.gif观察jvisualvm,发现产生20个NonRegisteringDriver 对象

a98328b87f4c48d3b44670f231eaa59a.gif

观察 hikari日志,确认有20个链接对象生成,以及产生总链接10个,空闲链接10个。

a98328b87f4c48d3b44670f231eaa59a.gif

初步判断一个数据库链接会生成两个 NonRegisteringDriver 对象。

启动压测程序,压测1000s

期间观察gc日志,gc时间间隔约20s,100s后发生5次 gc

a98328b87f4c48d3b44670f231eaa59a.gif

观察 hikari日志,确认有20个链接对象生成

a98328b87f4c48d3b44670f231eaa59a.gif观察jvisualvm变成 40个 NonRegisteringDriver 对象,符合预期。

a98328b87f4c48d3b44670f231eaa59a.gif

持续观察,1000s后理论上会产生220个对象(20 + 20 * 1000s / 100s),查看 jvisualvm 以下

a98328b87f4c48d3b44670f231eaa59a.gif产生了240个对象,基本和预期符合。

实验结果分析

再结合咱们生产的问题,假设咱们天天14个小时高峰期(12:00 ~ 凌晨2:00),期间链接数20,10个小时低峰期,期间链接数10,每次 full gc 间隔14天,等到下次 full gc 堆积的 NonRegisteringDriver 对象为 (20 * 14 + 10 * 10) * 2 * 14 = 10640,与问题dump里面NonRegisteringDriver对象的数量10140 个基本吻合。

至此问题根源已经获得彻底确认!!!

4、问题解决方案

由上面分析可知,问题产生的废弃的数据库链接对象堆积,最终致使 full gc 时间过长。因此咱们能够从如下方面思考解决方案:

一、减小废弃的数据链接对象的产生和堆积。

二、优化full gc时间.

a98328b87f4c48d3b44670f231eaa59a.gif

【调整hikari参数】

咱们能够考虑设置 maxLifetime 为一个较大的值,用于延长链接的生命周期,减小产生被废弃的数据库链接的频率,等到下次 full gc 的时候须要清理的数据库链接对象会大大减小。

Hikari 推荐 maxLifetime 设置为比数据库的 waittimeout 时间少 30s 到 1min。若是你使用的是 mysql 数据库,可使用 show global variables like '%timeout%'; 查看 waittimeout,默认为 8 小时。

下面开始验证,设置maxLifetime = 1小时,其余条件不变。压测启动前观察jvisualvm,NonRegisteringDriver 对象数量为20

a98328b87f4c48d3b44670f231eaa59a.gif1000s,观察 NonRegisteringDriver 对象仍然为20

a98328b87f4c48d3b44670f231eaa59a.gifNonRegisteringDriver 对象没有发生堆积,问题获得解决。

同时另外注意:minimumIdle和maximumPoolSize不要设置得太大,通常来讲配置minimumIdle=10,maximumPoolSize=10~20便可。

【使用G1回收器】

G1回收器是目前java垃圾回收器的最新成果,是一款低延迟高吞吐的优秀回收器,用户能够自定义最大暂停时间目标,G1会尽量在达到高吞吐量同时知足垃圾收集暂停时间目标。

下面开始验证G1回收器的实用性,该验证过程须要一段较长时间的观察,同时借助链路追踪工具skywalking。最终观察了10天,结果图以下:使用G1回收器,部分jvm参数-Xms3G -Xmx3G -XX:+UseG1GC

a98328b87f4c48d3b44670f231eaa59a.gif

使用java 8默认的Parallel GC回收器组合,部分jvm参数-Xms3G -Xmx3G

a98328b87f4c48d3b44670f231eaa59a.gif以上图中四个内容,从左到右分别为

一、堆内存,分为已使用和空闲内存。

二、方法区内存,这个不须要关注

三、young gc和full gc时间

四、程序启动之后young gc和full gc次数

咱们能够看到使用Parallel GC回收器组合的服务消耗的内存速度较快,发生了6996次young gc且发生了一次full gc,full gc时间长达5s。另一组使用G1回收器的服务消耗内存速度较为平稳,只发生3827次young gc且没有发生full gc。由此能够看到G1回收器确实能够用来解决咱们的数据库链接对象堆积问题。

【创建巡查系统】

这个咱们目前尚未通过实践,可是根据上面分析结果判断,按期触发full gc能够达到每次清理少许堆积的数据库链接的做用,避免过多数据库链接一直堆积。采用该方法须要对业务的内容和高低峰周期很是熟悉。实现思路参考以下:

一、建立java程序,使用定时任务按期调用System.gc()。该方法的缺点是即便手动调用了System.gc(),jvm不必定会马上开始回收工做,有可能会根据它自己的算法,自行选择最优时间才开始进行回收工做。

二、建立shell脚本调用jmap -dump:live,file=dump001.bin PID,使用linux的crontab任务保证定时执行,执行完后再把dump001.bin删掉便可。该方法能保证必定发生full gc,缺点是功能过于单一零散,很差集中管理。

5、总结

咱们此次问题产生的根源是数据库链接对象堆积,致使full gc时间过长。解决思路能够从如下三点入手:

一、调整hikari配置参数。例如把maxLifetime设置为较大的值(比数据库的wait_timeout少30s),minimumIdle和maximumPoolSize值不能设置太大,或者直接采用默认值便可。

二、采用G1垃圾回收器。

三、创建巡查系统,在业务低峰期主动触发full gc。

Java面试题专栏

a98328b87f4c48d3b44670f231eaa59a.gif

欢迎长按下图关注公众号后端技术精选

a98328b87f4c48d3b44670f231eaa59a.gif

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值