jhat(Java Heap Analyse Tool)
用途:是用来分析java堆的命令,可以将堆中的对象以html的形式显示出来,包括对象的数量,大小等等,并支持对象查询语言
使用:
| [root@www wangxiaoxiao]# jhat heap.map #heap.map是通过jmap -dump:file=heap.map pid命令导出
Reading from heap.map...
Dump file created Tue Feb 14 10:50:34 CST 2017
Snapshot read, resolving...
Resolving 2570075 objects...
Chasing references, expect 514 dots
...........................................................................................................................
................................................................................................................
Eliminating duplicate references
............................................................................................................................
...............................................................................................................
...............................................................................................................
Snapshot resolved.
Started HTTP server on port 7000
Server is ready. |
访问 http://localhost:7000,就可以查看详细的内存信息
关于jhat,不打算多说,因为一般实际开发中,很少使用jhat来直接对内存dump文件进行分析。下面介绍一个MAT,这个用的比较多。
MAT(Memory Analyzer Tool)
MAT是一个eclipse的插件,上手起来比较快。它能够快速的分析dump文件,可以直观的看到各个对象在内存占用的量大小,以及类实例的数量,对象之间的引用关系,找出对象的GC Roots相关的信息,此外还能生成内存泄露报表,疑似泄露大对象的报表等等。
安装MAT
可以选择eclipse插件的方式安装
http://download.eclipse.org/mat/1.3/update-site/
也可以选择单独MAT程序下载安装
http://www.eclipse.org/mat/downloads.php
内存溢出时,自动保存dump文件
前面是手工导出内存dump映射文件,如果应用已经在线上运行,为了能获取应用出现内存溢出时获得heap dump文件,以便在之后分析,可以在JVM启动时指定参数:-XX:+HeapDumpOnOutOfMemoryError,JVM会在遇到OutOfMemoryError时保存一个“堆转储快照”,并将其保存在一个文件中。 文件路径通过-XX:HeapDumpPath指定。
案例
<pre class="prettyprint linenums lang-java prettyprinted" style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; display: block; padding: 9.5px; margin: 0px 0px 10px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; overflow-wrap: break-word; background-color: rgb(51, 51, 51); border: 1px solid rgb(204, 204, 204); border-radius: 0px !important; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: start; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; text-decoration-style: initial; text-decoration-color: initial;">
- public class OomDemo {
- public static void main(String[] args) {
- Map<String,Long> map=new HashMap<String, Long>();
- for (long i = 0; i < Long.MAX_VALUE; i++) {
- map.put(i+"",i);
- }
- }
- }
设置虚拟机参数为:-Xmx40m -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=D:\Java\dump.hprof
执行程序,很快会抛出异常:
java.lang.OutOfMemoryError: GC overhead limit exceeded
Dumping heap to D:\Java\dump ...
Exception in thread "main" java.lang.OutOfMemoryError: GC overhead limit exceeded
Heap dump file created [55940678 bytes in 0.556 secs]
at java.lang.StringBuilder.toString(StringBuilder.java:405)
at oom.OomDemo.main(OomDemo.java:13)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:606)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:144)
生成了文件之后,就可以通过MAT打开来进行分析
File->Open Heap Dump->选择文件
弹出如下界面:
选中第一个,点击finish,出现以下界面:
可以看到,提示在主线程(main)的一个本地变量中使用了98.77%的内存,而这个本地变量就是java.util.HashMap$Entry[]。