最近通过监控发现java服务的堆内存每天都会上涨150MB左右,通过分析监控发现是堆内存中的PS Old Gen(老生代)区域每天上涨导致堆内存上涨,而且也不会被回收; 本篇文章就带领大家通过MAT(Memory Analyzer Tool)内存分析工具一起找出老生代内存长涨的对象
MAT(Memory Analyzer Tool)内存分析工具
点击【官方下载】进行下载;
MAT依赖本地JAVA_HOME环境变量设置,如果本地JAVA_HOME是JDK8的话,直接将JDK11下载到MAT安装目录,并将JDK11的路径通过-vm参数添加到MemoryAnalyzer.ini配置文件中
JDK11 MAT依赖最低JDK11,请大家自行前往oracle官方网站下载;
JVisualvm、Jmap或【阿里开源的java分析工具Arthas】;
vjmap
一款开源的增强版jmap,用于输出老生代内存区的内存地址,【下载地址】
可以下载vjtools,通过maven编译vjmap,通过maven的install命令会在目标目录生成vjmap.zip,里面是完整的运行环境
# ps -aef | grep java
如下图所示:
3. JVisualVM可以显示java服务的PID
备注: 以上3种方式选一种即可
java -jar arthas-boot.jar
如下图所示:
[arthas]$ heapdump
说明:上面是导出堆内存信息的两种方式,选一种即可,需复制下hprof文件的导出路径,MAT会用到。
vjmap.bat -address 15832
Linux的运行vjmap.sh -address PID命令即可
输出信息:
PSYoungGen [
eden = [0x00000007d5560000,0x00000007db9e1478,0x00000007f3250000] ,
from = [0x00000007f3250000,0x00000007f3be8000,0x00000007f9920000] ,
to = [0x00000007f9a40000,0x00000007f9a40000,0x0000000800000000] ]
PSOldGen [ [0x0000000780000000,0x00000007846715f8,0x00000007d5560000] ]
由此可以得出老生代内存地址范围为PSOldGen [ [0x0000000780000000,0x00000007846715f8,0x00000007d5560000] ],其中
0x0000000780000000:PSOldGen的起始地址
0x00000007846715f8:表示PSOldGen已使用了的结束地址;
0x00000007d5560000:表示PSOldGen的结束地址;
打开刚才导出的hprof文件
进入OQL编辑画面,执行查询老生代内存区域中的对象
查询OQL语句如下:
SELECT *
FROM INSTANCEOF java.lang.Object t
WHERE toHex(t.@objectAddress)>="0x780000000" AND toHex(t.@objectAddress)<="0x7d5560000"
- 注: 上面oql语句条件中的objectAddress中的值是通过vjmap工具输出信息中的PSOldGen得出的,记得使用时要删除0x后面的00000;
- 上图中的shallow Heap是指对象在没有引用其它对象、数组时,对象本身的大小;
- 上图中的Retained Heap是指对象本身的大小+它所引用对象的大小,即对象的shallow Heap+该对象所引用的所有对象大小的总和,即GC要回收该对象的内存大小
到这一步,剩下的工作就是分析上图所列出的对象,分析其GC路径和引用路径,分析程序中产生内存泄露的代码了;
本文发布于:2024-02-01 06:39:42,感谢您对本站的认可!
本文链接:https://www.4u4v.net/it/170674078234617.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
留言与评论(共有 0 条评论) |