一、先上一段oom代码
先调整一下jvm参数:
-Xms10m -Xmx10m -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=C:/Users/cxp/Desktop/heap.hprof -XX:+PrintGCDetails -XX:+PrintGCTimeStamps
- -Xms10m:堆初始大小10m
- -Xmx10m:堆最大10m
- -XX:+HeapDumpOnOutOfMemoryError:收集oom异常堆信息
- -XX:HeapDumpPath=C:/Users/cxp/Desktop/heap.hprof:输出指定文件
- -XX:+PrintGCDetails:打印gc详情
- -XX:+PrintGCTimeStamps:打印gc时间戳
public class Oom {
public static void main(String[] args) {
List<User> userList = new ArrayList<>();
while (true) {
userList.add(new User());
}
}
}
运行一下,就会报错:
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
at java.util.Arrays.copyOf(Arrays.java:3210)
at java.util.Arrays.copyOf(Arrays.java:3181)
at java.util.ArrayList.grow(ArrayList.java:267)
at java.util.ArrayList.ensureExplicitCapacity(ArrayList.java:241)
at java.util.ArrayList.ensureCapacityInternal(ArrayList.java:233)
at java.util.ArrayList.add(ArrayList.java:464)
at com.cxp.study.test.Oom.main(Oom.java:24)
同时在我们刚刚配置的路径下(-XX:HeapDumpPath=?)看到输出的堆异常文件:*.hprof
二、分析堆异常文件heap.hprof
1.先用jhat试试:
PS C:\Users\cxp\Desktop> jhat .\heap.hprof
Reading from .\heap.hprof...
Dump file created Tue Sep 14 17:54:10 CST 2021
Snapshot read, resolving...
Resolving 172243 objects...
Chasing references, expect 34 dots..................................
Eliminating duplicate references..................................
Snapshot resolved.
Started HTTP server on port 7000
Server is ready.
然后浏览器访问 http://localhost:7000/ ,看到:
最后可以看到我们对象创建过多,发生oom了,但是,看不到在哪发生的,具体是怎么发生的,感觉不是太友好,不知道是不是有方式可以找到,求教一下。
2.用JVisualVM来分析
windows10控制台输入JVisualVM命,打开窗口:
点击 文件=》装入,文件类型选择*.hprof,选择我们生成的heap.hprof文件,看到:
可以清晰看到堆信息和oom异常发生的线程,点击【导致 OutOfMemoryError 异常错误的线程:】后面的 【main 】进入:
可以清楚看到发生异常的位置,感觉很nice。
最后还是推荐大家使用JVisualVM,很友好。