java程序内存泄漏怎么排查
- 首先了解几个命令
- 怎么判断当前程序有没有出现内存溢出
- 模拟代码
- 模拟步骤
- 判断依据
- 出现内存溢出怎么办
- 最原始的方法
- 使用JProfiler解析hprof文件
- 在线dump文件分析网站https://console.heapdump.cn/
- MAT
- 第二种方案
- 总结
上文提到面试碰到CPU占满时的问题解决,决心把内存泄漏的问题也一起看一看,如果有更好的方案,评论区请指教。
首先了解几个命令
pId 为程序id
# 获取当前linux上部署的所有java程序 pid和运行占有资源情况(俩个都可以)
ps -aux|grep java
top $(ps -e | grep java | awk '{print $1}' | sed 's/^/-p/')
#获取某个java程序所有线程资源占用情况
top -Hp [pId]
# 获取某个java程序的线程执行情况(会展示在哪个类的哪一行)
jstack [pId]
# 获取某个java程序的gc情况
jstat -gcutil [pId]
# 获取某个java程序的gc情况 每秒获取一次 获取10次
jstat -gcutil [pId] 1000 10
# 将某个java程序的heap(堆)情况写入heap.hprof文件
jmap -dump:format=b,file=heap.hprof [pId]
怎么判断当前程序有没有出现内存溢出
模拟代码
ps: 模拟代码大概要执行10分钟才会溢出,可以加入多个String多个集合实现快些。
import java.util.ArrayList;
import java.util.List;
/**
* @Author 打工人
* @Date 2022/4/4 下午9:14
* @Version 1.0
* @Remarks
*/
public class SystemOOMTest {
public static final String SYSTEM_OOMTEST_STR = "SYSTEM_OOMTEST_STR";
public static void main(String[] args) {
int i = 0;
String thisStr;
List<String> oomObjs = new ArrayList<String>();
while (true){
thisStr = SYSTEM_OOMTEST_STR + i++;
oomObjs.add(thisStr);
try {
Thread.sleep(100L);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(thisStr);
}
}
}
模拟步骤
jvm初始和最大内存设置2M运行
javac SystemOOMTest2.java
java -Xms2M -Xmx2M SystemOOMTest2
判断依据
每10秒展示一次gc 获取10次
jstat -gcutil 13869 10000 10
jvm的对象创建机制为先存储Edan区(大对象会直接存储在Old区),存满后,执行YGC,没有被GC到的对象复制到S0或者S1区,都存满还是没有可用内存的话将S区的对象存储在O(Old)区,Old区存满之后执行FGC(FullGC)。
java程序内存溢出判断的大依据为FGC的次数,如果FGC次数很频繁且每次GC完释放的内存很少就有可能是出现了内存溢出。(为什么是有可能呢,因为也有情况是内存确实不够用)
出现内存溢出怎么办
最原始的方法
将java程序的堆占用情况输出到heap.hprof文件中,然后使用解析工具进行解析,解析完成之后查看实例占用多的类。然后进入代码分析,查看为什么没有被回收。
format=b:表示生成二进制类型的dump文件
PS: 执行下面语句jvm会执行一次FGC,并且会造成程序挂起(线上慎用)。文件大的话在解析时也会很慢。
jmap -dump:format=b,file=heap.hprof [pId]
使用JProfiler解析hprof文件
在线dump文件分析网站https://console.heapdump.cn/
MAT
楼主最开始也尝试的这个工具,不过 MAC上面安装的1.8.0版本的,打开解析面板是空白的。
第二种方案
最好能在本地复盘。通过查看线上日志,和近期的发版内容,用户增量一起结合起来看。毕竟线上真遇见了,堆内存占用会很大,在写出的时候java程序会挂起,影响运行。
代码层面最好避免这种情况,使用完的对象和集合要确定会不会被jvm正常GC掉。
PS:堆中的实例对象GC依据为 当前实例对象还有无地方引用。
总结
- 定位到内存占用过高的java程序pid
- 查看java程序的gc情况
- 如果FGC次数很频繁 导出一份heap数据
- 使用解析工具解析heap