在本文中,我们将展示用Java捕获堆转储的不同方法。
堆转储是JVM内存中某一时刻所有对象的快照。它们对于解决内存泄漏问题和优化Java应用程序中的内存使用非常有用。
堆文件通常以二进制格式存储。我们可以使用jhat或JVisualVM之类的工具打开和分析这些文件。另外,对于Eclipse用户来说,使用MAT是非常常见的。
下面我们将介绍生成堆转储的多种工具和方法,并将展示它们之间的主要区别。
JDK工具
JDK附带了几种工具,可以以不同的方式捕获堆转储。所有这些工具都位于JDK文件夹下的主目录中。因此,只要这个目录包含在系统路径中,我们就可以从命令行启动它们。
Jmap
jmap是一种打印运行JVM内存统计信息的工具。我们可以将其用于本地或远程进程。
要使用jmap捕获堆转储,我们需要使用dump选项:
jmap-dump:[live],格式=b,file=<file path><pid>
除了这个选项,我们还应该指定几个参数:
- live:如果设置,则只打印具有活动引用的对象,并丢弃准备进行垃圾收集的对象。此参数是可选的
- format=b:指定转储文件将采用二进制格式
- file:将写入转储的文件
- pid:Java进程的id
例如:
jmap -dump:live,format=b,file=/tmp/dump.hprof 12587
我们可以通过使用jps命令轻松获得Java进程的pid。
请记住,jmap是作为实验工具引入JDK中的,它不受支持。因此,在某些情况下,最好使用其他工具。
Jcmd
jcmd是一个非常完整的工具,它通过向JVM发送命令请求来工作。我们必须在运行Java进程的同一台机器上使用它。
它的许多命令之一是GC.heap_dump. 我们只需指定进程的pid和输出文件路径,就可以使用它获取堆转储:
jcmd <pid> GC.heap_dump <file-path>
我们可以在执行之前使用相同的参数:
jcmd 12587 GC.heap_dump /tmp/dump.hprof
与jmap一样,生成的转储是二进制格式的。