JVM常用命令之-----jhat

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指定。

案例

  1. public class OomDemo {
  2. public static void main(String[] args) {
  3. Map map=new HashMap();
  4. for (long i = 0; i < Long.MAX_VALUE; i++) {
  5. map.put(i+"",i);
  6. }
  7. }
  8. }

设置虚拟机参数为:-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->选择文件
弹出如下界面:


JVM常用命令之-----jhat_第1张图片
QQ截图20170214134508.png

选中第一个,点击finish,出现以下界面:

JVM常用命令之-----jhat_第2张图片
QQ截图20170214134727.png

可以看到,提示在主线程(main)的一个本地变量中使用了98.77%的内存,而这个本地变量就是java.util.HashMap$Entry[]。

你可能感兴趣的:(JVM常用命令之-----jhat)