mysql导致的内存泄漏Abandoned connection cleanup thread

今天突然收到了报警,说内存占用超过了95%,赶紧到服务器看下

top一下,发现SpringBoot占用内存过大

jmap看一下是什么占用了内存

bash 复制代码
ps -ef | grep java
jmap -histo:live 31021 | head -30

排名前3占用特别大,第一个是[C,表示char[]数组,第二个Constructor是用来构造反射的,第三个Class是加载的元数据,大概意思是通过反射大量创建了对象。

奇怪了, java不是有垃圾回收机制吗,为什么还会创建这么多对象,带着疑问先dump下来再说

bash 复制代码
jmap -dump:live,format=b,file=heap_31021.hprof 31021

拿到dump文件,之后需要通过EclipseMAT内存分析工具打开,下载地址:Eclipse MAT

安装完Exclipse MAT后,打开文件,选择刚刚dump的heap_31021.hprof文件

不要着急,需要等待一两分钟,如果显示这个,直接finish即可

进入之后,可以看到有个泄漏疑点的页签

如果没有,在首页也可以进入

看到了3个疑点

划线部分的Abandoned connection cleanup thread 是MySQL的一个清理线程,用来清理残留的连接,LaunchedURLClassLoader是SpringBoot的类加载器

大概意思就是MySQL的清理线程一直吊着ClassLoader,导致大量通过ClassLoader创建之后的类和对象无法回收

说人话,就是你代码里用到的反射,基本都要用ClassLoader创建,而MySQL一直吊着ClassLoader,导致GC回收的时候,发现CladdLoader还被引用着,所以通过ClassLoader加载的类和对象,无法被回收和释放

进一步观察,我们去找找MySQL源码,直接搜AbandonedConnectionCleanupThread,如图所示,有没有觉得这串英文很熟悉

MySQL启动了个线程在后台跑

而且还通过静态代码块启动,不归Spring管理,就会导致线程一直无法正常关闭,ClassLoader一直被引用

但这并不是直接内存溢出的原因,一般是由这几个原因并发构成的:

1.用的mysql驱动包是5.x版本

2.项目里大量用到反射

3.项目并发高,并且持续运行了一段时间

解决方式

Spring关闭的时候,销毁一下MySQL的清理线程

java 复制代码
@Component
public class MysqlCleanup {
    @PreDestroy
    public void shutdown() throws Exception {
        AbandonedConnectionCleanupThread.checkedShutdown();
    }
}

升级MySQL驱动版本到8.x

XML 复制代码
<dependency>
    <groupId>com.mysql</groupId>
    <artifactId>mysql-connector-j</artifactId>
    <version>8.0.33</version>
</dependency>

8.x版本驱动完美兼容mysql5.x版本的mysql,所以不用担心,只需要连接信息做一点小改造就好

driver-class-name原本是com.mysql.jdbc.Driver,改成com.mysql.cj.jdbc.Driver

8.x版本有时区问题,url连接参数加一个&serverTimezone=Asia/Shanghai就好了

相关推荐
WangYaolove131442 分钟前
基于python的在线水果销售系统(源码+文档)
python·mysql·django·毕业设计·源码
山岚的运维笔记1 小时前
SQL Server笔记 -- 第18章:Views
数据库·笔记·sql·microsoft·sqlserver
roman_日积跬步-终至千里2 小时前
【LangGraph4j】LangGraph4j 核心概念与图编排原理
java·服务器·数据库
汇智信科2 小时前
打破信息孤岛,重构企业效率:汇智信科企业信息系统一体化运营平台
数据库·重构
野犬寒鸦2 小时前
从零起步学习并发编程 || 第六章:ReentrantLock与synchronized 的辨析及运用
java·服务器·数据库·后端·学习·算法
霖霖总总2 小时前
[小技巧66]当自增主键耗尽:MySQL 主键溢出问题深度解析与雪花算法替代方案
mysql·算法
晚霞的不甘4 小时前
揭秘 CANN 内存管理:如何让大模型在小设备上“轻装上阵”?
前端·数据库·经验分享·flutter·3d
市场部需要一个软件开发岗位4 小时前
JAVA开发常见安全问题:纵向越权
java·数据库·安全
海奥华24 小时前
mysql索引
数据库·mysql
2601_949593655 小时前
深入解析CANN-acl应用层接口:构建高效的AI应用开发框架
数据库·人工智能