写点什么

java 培训:内存泄漏问题排查与分析

作者:@零度
  • 2022 年 3 月 08 日
  • 本文字数:910 字

    阅读完需:约 3 分钟

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)添加配置参数,限制缓存大小



参数解释:

https://docs.jboss.org/hibernate/orm/5.4/userguide/html_single/Hibernate_User_Guide.html#configurations-query


(2)提高 sql in 的缓存效率



参数解释:

https://docs.jboss.org/hibernate/orm/5.4/userguide/html_single/Hibernate_User_Guide.html#configurations-query



文章来源于 Java 编程

用户头像

@零度

关注

关注尚硅谷,轻松学IT 2021.11.23 加入

还未添加个人简介

评论

发布
暂无评论
java培训:内存泄漏问题排查与分析_JAVA开发_@零度_InfoQ写作平台