java 培训:内存泄漏问题排查与分析
1、怀疑内存泄漏
进入 APM 监控系统查看实例内存情况,把时间线拉长到一天,可以看到内存有缓慢上升趋势,初步怀疑有内存泄漏。
2、Heap Dump
获取到机器 ip,联系运维人员去机器上把堆 dump 下来,dump 命令:
/xxx/jdk1.8.0_212/bin/jmap -dump:live,format=b,file=/xxx/xxx.hprof 进程号
将堆 dump 文件 Xxx.hprof 下载到本地。
3、下载 Heap Dump 分析工具
常用的分析工具有 MAT 和 JProfile,本文以 MAT 工具为示例进行分析。工具下载链接如下:
https://www.eclipse.org/mat/downloads.php
注意:如果你本地安装的是 JDK11+,下载最新java培训的即可;如果你本地安装的是 JDK8,建议下载 1.9.2 版本。
4、将 Dump 文件导入 MAT 工具
MAT 是 eclipse 的一个插件,免安装,双击打开即可使用。
打开下载好的 dump 文件
5、分析 Dump 文件
打开内存泄漏怀疑分析报告,可以看到 SessionFactoryImpl 这个对象使用了 149M 内存,占总内存的 25%,这肯定不正常。
SessionFactoryImpl 这个类是跟数据库相关,服务代码中使用了 JPA 作为持久层框架,应该是跟 JPA 相关,继续往下分析。
我们打开内存树,往下挖,可以看到 QueryPlanCache 这个对象占用内存比较大。
QueryPlanCache 表面的意思是:查询计划缓存。用 google 查一下具体含义。
简单来说 Hibernate 会缓存 sql 语句以减少重复编译,便于直接命中提高效率。
在使用 SQL in 的时候,如果 in 后的参数不同,hibernate 会把其当成不同的 sql 进行缓存,从而缓存大量的 sql。
缓存的大小是多少?查了一下官方文档,如果不配置,这个缓存默认最大值为 2048 。
Stack Overflow 上也有用户反馈这个问题:
https://stackoverflow.com/questions/31557076/spring-hibernate-query-plan-cache-memory-usage
7、分析结论
drawio 服务里面有大量的 sql in 语句,in 后面的参数不一样造成 Hibernate 缓存了大量 SQL 语句,占用大量的堆内存。
8、解决措施
(1)添加配置参数,限制缓存大小
参数解释:
(2)提高 sql in 的缓存效率
参数解释:
文章来源于 Java 编程
评论