Java 与 C++ 之间有一堵由内存动态分配和垃圾收集技术所围成的“高墙”,墙外面的人想进去,墙里面的人却想出来。
文章目录
一、概述
给一个系统定位问题的时候,知识经验是关键基础,数据是依据,工具是运用知识处理数据的手段。这里说的数据包括:
运行日志、异常堆栈、GC日志、线程快照(threaddump/javacore文件)、堆转储快照(heapdump/hprof文件)
等。经常使用适当的虚拟机监控和分析的工具可以加快我们分析数据、定位解决问题的速度,但学习工具前,也应当意识到工具永远都是知识技能的一层包装,不可能学会了就能包治百病。
二、JDK 的命令行工具
JDK 的 bin 目录下包含了所有命令行工具,这里将介绍其中的一部分,包括用于
监视虚拟机和故障处理
的工具。这些工具都非常稳定而且功能强大,能在处理应用程序性能问题、定位故障时发挥很大作用。
- JDK 主要命令行监控工具的用途
名称 | 主要作用 |
---|---|
jps | JVM Process Status Tool,显示指定系统内所有的 HotSpot 虚拟机进程 |
jstat | JVM Statistics Monitoring Tool,用于收集 HotSpot 虚拟机各方面的运行数据 |
jinfo | Configuration Info for Java,显示虚拟机配置信息 |
jmap | Memory Map for Java,生成虚拟机的内存转储快照(heapdump 文件) |
jhat | JVM Heap Dump Browser,用于分析 heapdump 文件,它会建立一个 HTTP/HTML 服务器,让用户可以在浏览器上查看分析结果。 |
jstack | Stack Trace for Java,显示虚拟机的线程快照 |
jps:虚拟机进程状况工具
- jps(JVM Process Status Tool),
可以列出正在运行的虚拟机进程,并显示虚拟机执行主类(main函数所在的类)名称以及这些进程的本地虚拟机唯一ID(Local Virtual Machine Identifier,LVMID)
。很多时候会使用这个命令查询 LVMID ,来确定要监控的是哪一个虚拟机进程。 - 命令格式:
jps [ options ] [ hostid ]
- hostid 为 RMI 注册表中注册的主机名。
- options 主要选项:
选项 | 作用 |
---|---|
-q | 只输出LVMID,省略主类的名称 |
-m | 输出虚拟机进程启动时传递给主类 main() 函数的参数 |
-l | 输出主类的全名,如果进程执行的是 Jar 包,输出 Jar 路径 |
-v | 输出虚拟机进程启动时 JVM 参数 |
- 演示
jstat:虚拟机统计信息监视工具
jstat(JVM Statistics Monitoring Tool),用于监视虚拟机各种运行状态信息
的命令行工具。它可以显示本地或者远程虚拟机进程中的类装载、内存、垃圾收集、JIT 编译等运行数据
,在没有 GUI 图形界面,只提供了纯文本控制台环境的服务器上,它将是运行期定位虚拟机性能问题的首选工具。
命令格式:jstat [ option vmid [interval[s|es] [count]] ]
- 对于命令中的 VMID 与 LVMID ,如果是本地虚拟机进程,这两者是一致的,如果是远程虚拟机进程,那 VMID 的格式应当是:
[protocol:][//]lvmid[@hostname[:port]/servername]
。 - 参数 interval 和 count 代表查询间隔和次数,如果省略这两个参数,说明只查询一次。假设需要每 250 毫秒查询一次进程 2764 垃圾收集状况,一共查询 20 次,那命令是:
jstat -gc 2764 250 20
。 - 选项 option 代表着用户希望查询的虚拟机信息,主要分为 3 类:
类装载、垃圾收集、运行期编译状况
,具体选项及作用如下表:
选项 | 作用 |
---|---|
-class | 监视类装载、卸载数量、总空间以及类装载所耗费的时间 |
-gc | 监视 Java 堆状况,包括 Eden 区、两个 survivor 区、老年代、永久代等的容量、已用空间、GC 时间合计等信息 |
-gccapacity | 监视内容与 -gc 基本相同,但输出主要关注 Java 堆各个区域使用到的最大、最小空间 |
-gcutil | 监视内容与 -gc 基本相同,但输出主要关注已使用空间占总空间的百分比 |
-gccause | 与 -gcutil 功能一样,但是会额外输出导致上一次 GC 产生的原因 |
-gcnew | 监视新生代 GC 状况 |
-gcnewcapacity | 监视内容与 -gcnew 基本相同,输出主要关注使用到的最大、最小空间 |
-gcold | 监视老年代 GC 状况 |
-gcoldcapacity | 监视内容与 -gcold 基本相同,输出主要关注使用到的最大、最小空间 |
-gcpermcapacity | 输出永久代使用到的最大、最小空间 |
-compiler | 输出 JIT 编译器编译过的方法、耗时等信息 |
-printcompilation | 输出已经被 JIT 编译的方法 |
演示
各参数说明:
- S0C:年轻代中 To Survivor 的容量(单位 KB);
- S1C:年轻代中 From Survivor 的容量(单位 KB);
- S0U:年轻代中 To Survivor 目前已使用空间(单位 KB);
- S1U:年轻代中 From Survivor 目前已使用空间(单位 KB);
- EC:年轻代中 Eden 的容量(单位 KB);
- EU:年轻代中 Eden 目前已使用空间(单位 KB);
- OC:Old 代的容量(单位 KB);
- OU:Old 代目前已使用空间(单位 KB);
- MC:Metaspace 的容量(单位 KB);
- MU:Metaspace 目前已使用空间(单位 KB);
- YGC:从应用程序启动到采样时年轻代中 gc 次数;
- YGCT:从应用程序启动到采样时年轻代中 gc 所用时间 (s);
- FGC:从应用程序启动到采样时 old 代(全 gc)gc 次数;
- FGCT:从应用程序启动到采样时 old 代(全 gc)gc 所用时间 (s);
- GCT:从应用程序启动到采样时 gc 用的总时间 (s)。
jinfo:Java 配置信息工具
- jinfo(Configuration Info for Java),其作用是
实地查看和调整虚拟机各项参数
。 - 命令格式:
jinfo [ option ] pid
- JDK 1.6 中,jinfo 对于 Windows 平台功能仍然有较大限制,只提供了最基本的
-flag
选项。
jmap:Java 内存映像工具
- jmap(Memory Map for Java),用于
生成堆转储快照(一般称为 heapdump 或 dump文件)
。如果不使用 jmap 命令,要想获取 Java 堆转储快照,还有一些比较暴力的手段:- 在前面用过的
-XX:+HeapDumpOnOutOfMemoryError
参数,可以让虚拟机在 OOM 异常出现之后自动生成 dump 文件。 - 可以使用
-XX:HeapDumpPath
指定 dump 文件导出路径或文件名 - 通过
-XX:+HeapDumpOnCtrlBreak
参数则可以使用 Ctrl + Break 键让虚拟机生成 dump 文件。 - 或者在 Linux 系统下通过
Kill -3
命令发送进程退出信号“吓唬”一下虚拟机,也能拿到 dump 文件。
- 在前面用过的
- jmap 的作用不仅仅是为了获取 dump 文件,它还可以
查询 finalize 执行队列、Java 堆和永久代的详细信息
,如空间使用率、当前用的是哪种收集器等。 - 命令格式:
jmap [ option ] vmid
,option 选项如下表:
选项 | 作用 |
---|---|
-dump | 生成 Java 堆转储快照。格式为:-dump:[live, ]fomat=b, file=<filename> ,其中live子参数说明是否只 dump 出存活的对象 |
-finalizerinfo | 显示在 F-Queue 中等待 Finalizer 线程执行 finalize 方法的对象。只在 Linux/Solaris 平台下有效 |
-heap | 显示Java堆详细信息,如使用哪种回收器、参数配置、分代状况等。只在 Linux/Solaris 平台下有效 |
-histo | 显示堆中对象统计信息,包括类、实例数量、合计容量 |
-permstat | 以 ClassLoader 为统计口径显示永久代内存状态。只在 Linux/Solaris 平台下有效 |
-F | 当虚拟机进程对 -dump 选项没有响应时,可使用这个选项强制生成 dump 快照。只在 Linux/Solaris 平台下有效 |
- 演示
jhat:虚拟机堆转储快照分析工具
- Sun JDK 提供
jhat 命令与 jmap 搭配使用,来分析 jmap 生成的堆转储快照
。 - 一般情况不会使用这个工具,功能较简陋,可以使用 VisualVM,或者 MAT 分析工具。
jstack:Java 堆栈跟踪工具
jstack 命令用于生成虚拟机当前时刻的线程快照
。线程快照就是当前虚拟机内每一条线程正在执行的方法堆栈的集合,生成线程快照的主要目的是定位线程出现长时间停顿的原因
,如线程间死锁、死循环、请求外部资源导致的长时间等待等都是导致线程长时间停顿的常见原因。线程出现停顿的时候通过 jstack 来查看各个线程的调用堆栈,就可以知道没有响应的线程到底在后台做些什么事,或者等待着什么资源。
HSDIS:JIT 生成代码反汇编
HSDIS 是一个 Sun 官方推荐的 HotSpot 虚拟机 JIT 编译代码的反汇编插件
,它包含在 HotSpot 虚拟机源码之中,但没有提供编译后的程序。
三、JDK 的可视化工具
JConsole:Java 监视与管理控制台
VisualVM:多合一故障处理工具
笔记来源:《深入理解Java虚拟机》第四章 虚拟机性能监控与故障处理工具(P101)。