JVM——性能监控与故障处理工具
HotSpot虚拟机提供了许多虚拟机运行时调试工具,方便开发人员在应用运行时对虚拟机和应用性能进行监控和优化
1 JDK命令行工具
| 名称 | 全称 | 用途 |
|---|---|---|
jps |
JVM Process Status Tool |
显示指定系统内所有的HotSpot虚拟机进程 |
jstat |
JVM Statistics Monitoring Tool |
用于收集Hotspot虚拟机各方面的运行数据 |
jinfo |
Configuration Info for Java |
显示虚拟机配置信息 |
jmap |
JVM Memory Map |
生成虚拟机的内存转储快照,生成heapdump文件 |
jhat |
JVM Heap Dump Browser |
用于分析heapdump文件,它会建立一个HTTP/HTML服务器,让用户在浏览器上查看分析结果 |
jstack |
JVM Stack Trace |
显示虚拟机的线程快照 |
1.1 jps:虚拟机进程状况工具
jps(JVM Process Status Tool)
- 列出正在运行的虚拟机进程
- 并显示虚拟机执行主类(Main Class,main()函数所在的类)的名称
- 以及这些进程的本地虚拟机的唯一ID(LVMID,Local Virtual Machine Identifier)
虽然功能比较单一,但它是使用频率最高的JDK命令行工具,因为其他JDK工具大多需要输入它查询到的LVMID来确定要监控的是哪一个虚拟机进程.对于本地虚拟机进程来说,LVMID与操作系统的进程ID(PID,Process Identifier)是一致的,使用Windows的任务管理器或Unix的ps命令也可以查询到虚拟机进程的LVMID,但如果使用了多个虚拟机进程,无法根据进程名称定位时,那就只能依赖jps命令显示主类的功能区才能区分了.
jps命令格式:
> jps [option] [hostid]
jps可以通过RMI协议查询开启了RMI服务的远程虚拟机进程状态,hostid为RMI注册表中注册的主机名.
其他常用选项
| 选项 | 作用 |
|---|---|
-q |
只输出LVMID,省略主类的名称 |
-m |
输出虚拟机进程启动时传递给主类的main()函数的参数 |
-l |
输出主类的全名,如果进程执行的是jar包,输出jar路径 |
-v |
输出虚拟机进程启动时JVM参数 |
1.2 jstat:虚拟机统计信息监控工具
jstat(JVM Statistics Monitoring Tool)是用于监控虚拟机各种运行状态信息的命令行工具.它可以显示本地或远程虚拟机进程中的类装载、内存、垃圾收集、JIT编译等运行数据,在没有GUI图像界面,只提高了纯文本控制台环境的服务器上,它将是运行期定位虚拟机性能问题的首选工具.
jstat命令格式:
> jstat [option vmid [interval[s|ms] [count]] ]
VMID与LVMID需要特别说明下:如果是本地虚拟机进程,VMID和LVMID是一致的,如果是远程虚拟机进程,那VMID的格式应当是:[protocol:][//] lvmid [@hostname[:port]/servername]
参数interval和count代表查询间隔和次数,如果省略这两个参数,说明只查询一次.
| 选项 | 作用 |
|---|---|
-class |
监视类装载、卸载数量、总空间及类装载所耗费的时间 |
-gc |
监视Java堆状况,包括Eden区、2个Survivor区、老年代、永久代等的容量 |
-gccapacity |
监视内容与-gc基本相同,但输出主要关注Java堆各个区域使用到的最大和最小空间 |
-gcutil |
监视内容与-gc基本相同,但输出主要关注已使用空间占总空间的百分比 |
-gccause |
与-gcutil功能一样,但是会额外输出导致上一次GC产生的原因 |
-gcnew |
监视新生代GC的状况 |
-gcnewcapacity |
监视内容与-gcnew基本相同,输出主要关注使用到的最大和最小空间 |
-gcold |
监视老年代GC的状况 |
-gcoldcapacity |
监视内容与——gcold基本相同,输出主要关注使用到的最大和最小空间 |
-gcpermcapacity |
输出永久代使用到的最大和最小空间 |
-compiler |
输出JIT编译器编译过的方法、耗时等信息 |
-printcompilation |
输出已经被JIT编译的方法 |
> jstat -gcutil 5828
S0 S1 E O P YGC YGCT FGC FGCT GCT
0.00 0.00 1.82 52.18 99.91 329 4.894 269 80.244 85.139
参数说明
- 新生代Eden区(E,表示Eden)使用了1.82%的空间
- 两个Survivor区(S0、S1,表示Survivor0、Survivor1)里面都是空的
- 老年代(O,表示Old)和永久代(P,表示Permanent)则分别使用了52.18%和99.91%的空间
- 程序运行以来共发生Minor GC(YGC,Young GC)329次
- 总耗时(YGCT,Young GC Time)4.894秒
- 发生Full GC(FGC)269次
- 总耗时(FGCT)80.244秒
- 所有GC总耗时(GCT)85.139秒
1.3 jinfo: Java配置信息工具
jinfo(Configuration Info for Java)的作用是实时地查看和调整虚拟机的各项参数.
使用jps的命令的-v参数可以查看虚拟机启动时显示指定的参数列表,但如果想知道未被显示指定的参数的系统默认值,除了去找资料外,就只能使用jinfo的-flag选项进行查询了(如果只限于JDK1.6或以上版本的话,使用java -XX:+PrintFlagsFinal查看参数默认值也是一个很好的选择),jinfo还可以使用-sysprops选项把虚拟机进程的System.getProperties()的内容打印出来.这个命令在JDK1.5时期已经随着Linux版的JDK发布,当时只提供了信息查询的功能,JDK1.6之后,jinfo在Windows和Linux平台都有提供,并且加入了运行期修改参数的能力,可以使用-flag[+|-]name或-flag name=valule修改一部分运行期可写的虚拟机参数值.JDK1.6中,jinfo对于Windows平台的功能仍然有较大的限制,只提供了最基本的-flag选项.
jinfo命令格式
> jinfo [option] pid
1.4 jmap: Java内存映像工具
jmap(Memory Map for Java)命令用于生产堆转储快照(一般称为heapdump或dump文件).它还可以查询finalize 执行队列,Java堆和永久代的详细信息,如空间使用率、当前用的是哪种收集器等.
jmap命令格式:
> jmap [option] vmid
| 选项 | 作用 |
|---|---|
-dump |
生成Java堆转储快照.格式为: -dump:[live,]format=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平台下有效 |
1.5 jhat:虚拟机堆转储快照分析工具
Sun JDK提供了jhat(JVM Heap Analysis Tool)命令与jmap搭配使用,来分析jmap生成的堆转储快照.
jhat内置了一个微型的HTTP/HTML服务器,生成dump文件的分析结果后,可以在浏览器中查看,不过实事求是地说,在实际工作中,除非真的没有别的工具可用,否则一般不会去直接使用jhat命令来分析demp文件,主要原因有二:意识一般不会在部署应用程序的服务器上直接分析dump文件,即使可以这样做,也会尽量将dump文件拷贝到其他机器上进行分析,因为分析工作时一个耗时且消耗硬件资源的过程,既然都要在其他机器上进行,就没必要收到命令行工具的限制了.另外一个原因是jhat的分析功能相对来说很简陋,VisualVM以及专门分析dump文件的Eclipse Memory Analyzer、IBM HeapAnalyzer等工具,都能实现比jhat更强大更专业的分析功能.
1.6 jstack: Java堆栈跟踪工具
jstack(Stack Trace for Java)命令用于生成虚拟机当前时刻的线程快照(一般称为threaddump或javacore文件).
线程快照就是当前虚拟机内每一条线程正在执行的方法堆栈的集合,生成线程快照的主要目的是定位线程出现长时间停顿的原因,如线程间死锁、死循环、请求外部资源导致的长时间等待等都是导致线程长时间停顿的常见原因.线程出现停顿的时候通过jstack来查看各个线程的调用堆栈,就可以知道没有响应的线程到底在后台做些什么事情,或者等待着什么资源.
jstack命令格式:
> jstack [option] vmid
| 选项 | 作用 |
|---|---|
-F |
当正常输出的请求不被响应时,强制输出线程堆栈 |
-l |
除堆栈外,显示关于锁的附加信息 |
-m |
如果调用到本地方法的话,可以显示C/C++的堆栈 |
2.jdk可视化工具
2.1 JConsole
JConsole工具在JDK/bin目录下,启动JConsole后,将自动搜索本机运行的jvm进程,不需要jps命令来查询指定.双击其中一个jvm进程即可开始监控,也可使用"远程进程"来连接远程服务器.
2.2 VisualVM
VisualVM是一个集成多个JDK命令行工具的可视化工具.VisualVM基于NetBeans平台开发,它具备了插件扩展功能的特性,通过插件的扩展,可用于显示虚拟机进程及进程的配置和环境信息(jps,jinfo),监视应用程序的CPU、GC、堆、方法区及线程的信息(jstat、jstack)等.VisualVM在JDK/bin目录下.