Java 堆转储 Dump 文件的几种方法,字节跳动技术面
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 一样,生成的转储是二进制格式的。
JVisualVM
=========
JVisualVM 是一个带有图形用户界面的工具,它允许我们监视、排除故障和分析 Java 应用程序。图形用户界面很简单,但非常直观,易于使用。
它的许多选项之一允许我们捕获堆转储。如果我们右键单击 Java 进程并选择“堆转储”选项,该工具将创建一个堆转储并在新选项卡中打开它:

注意,我们可以在“基本信息”部分找到创建的文件的路径。
从 jdk9 开始,visualvm 不包括在 oracle jdk 和 open jdk 发行版中。因此,如果我们使用 java9 或更新版本,我们可以从 visualvm 开源项目站点获得 JVisualVM。
自动捕获堆转储
=======
我们在前面几节中展示的所有工具都是为了在特定时间手动捕获堆转储。在某些情况下,我们希望在 java.lang.OutOfMemoryError 因此它有助于我们调查错误。
对于这些情况,Java 提供 HeapDumpOnOutOfMemoryError 命令行选项,当 java.lang.OutOfMemoryError 抛出:
java-XX:+HeapDumpOnOutOfMemoryError
默认情况下,它将转储存储在运行应用程序的目录下的 java_pid<pid>.hprof 文件中。如果要指定另一个文件或目录,可以在 HeapDumpPath 选项中进行设置:
java-XX:+HeapDumpOnOutOfMemoryError-XX:HeapDumpPath=<file or dir path>
当应用程序使用此选项耗尽内存时,我们将能够在日志中看到创建的包含堆转储的文件:
java.lang.OutOfMemoryError: Requested array size exceeds VM limit
Dumping heap to java_pid12587.hprof ...
Exception in thread "main" Heap dump file created [4744371 bytes in 0.029 secs]
java.lang.OutOfMemoryError: Requested array size exceeds VM limit
at com.baeldung.heapdump.App.main(App.java:7)
在上面的示例中,它被写入 java_pid12587.hprof 文件。
正如我们所看到的,这个选项非常有用,并且在使用这个选项运行应用程序时没有开销。因此,强烈建议始终使用此选项,特别是在生产中。
最后,还可以在运行时通过使用热点诊断 MBean 指定此选项。为此,我们可以使用 JConsole 并将 heapdumpOnAutoFMMemoryError VM 选项设置为 true:

JMX
===
本文将介绍的最后一种方法是使用 JMX。我们将使用上一节中简要介绍的热点诊断 MBean。此 MBean 提供接受 2 个参数的 dumpHeap 方法:
outputFile:转储文件的路径。该文件应具有 hprof 扩展名
live:如果设置为 true,它只转储内存中的活动对象,正如我们以前在 jmap 中看到的那样
在下一节中,我们将展示两种不同的方法来调用此方法以捕获堆转储。
JConsole
========
使用热点诊断 MBean 的最简单方法是使用 JConsole 等 JMX 客户机。
如果我们打开 JConsole 并连接到一个正在运行的 Java 进程,我们可以导航到 MBeans 选项卡并在下面找到热点诊断 com.sun.management 。在操作中,我们可以找到前面描述过的 dumpHeap 方法:

如图所示,我们只需要在 p0 和 p1 文本字段中引入参数 outputFile 和 live,以便执行 dumpHeap 操作。
程序化方式
=====
使用
《一线大厂 Java 面试题解析+后端开发学习笔记+最新架构讲解视频+实战项目源码讲义》
【docs.qq.com/doc/DSmxTbFJ1cmN1R2dB】 完整内容开源分享
热点诊断 MBean 的另一种方法是从 Java 代码以编程方式调用它。
为此,我们首先需要获取一个 MBeanServer 实例,以便获得在应用程序中注册的 MBean。之后,我们只需要获取一个热点 DiagnosticXbean 的实例并调用其 dumpHeap 方法。
让我们看看代码:
最后
我想问下大家当初选择做程序员的初衷是什么?有思考过这个问题吗?高薪?热爱?
既然入了这行就应该知道,这个行业是靠本事吃饭的,你想要拿高薪没有问题,请好好磨练自己的技术,不要抱怨。有的人通过培训可以让自己成长,有些人可以通过自律强大的自学能力成长,如果你两者都不占,还怎么拿高薪?
架构师是很多程序员的职业目标,一个好的架构师是不愁所谓的 35 岁高龄门槛的,到了那个时候,照样大把的企业挖他。为什么很多人想进阿里巴巴,无非不是福利待遇好以及优质的人脉资源,这对个人职业发展是有非常大帮助的。
如果你也想成为一名好的架构师,那或许这份 Java 核心架构笔记你需要阅读阅读,希望能够对你的职业发展有所帮助。
中高级开发必知必会:

评论