本文为《深入理解Java虚拟机+JVM高级特性与最佳实践》(第二版)读书笔记
1 命令行工具
1.1 jps
jps (JVM Process Status Tool)显示系统内所有的正在运行的Hotspot虚拟机进程,并显示虚拟机执行主类名称以及这些进程的本地虚拟机唯一ID
1.1.1 jps命令格式
jps [options] [hostid]
1.1.2 jps常用选项
-q | 只输出虚拟机ID,省略主类的名称 |
---|---|
-m | 输出虚拟机进程启动时传递给主类main() 函数的参数 |
-l | 输出主类的全名,如果进程执行的是jar包,输出jar包路径 |
-v | 输出虚拟机进程启动时JVM参数 |
1.2 jstat
jstat(JVM Statistics Monitoring Tool)是用于监视虚拟机各种运行状态信息的命令行工具。它可以显示本地或远程虚拟机进程中的类装载、内存、垃圾收集、JIT编译等运行数据,在没有GUI图形界面,只提供了纯文本控制台环境的服务器上,它将是运行期定位虚拟机性能问题的首选工具
1.2.1 jstat命令格式
jstat [option vmid [interval [s|ms] [count]]]
对于命令格式中的VMID与LVMID需要特别说明一下:如果是本地虚拟机进程,VMID与
LVMID是一致的,如果是远程虚拟机进程,那VMID的格式应当是:
[protocol:][//]lvmid[@hostname[:port]/servername]
参数interval和count代表查询间隔和次数,如果省略这两个参数,说明只查询一次。 假设
需要每250毫秒查询一次进程2764垃圾收集状况,一共查询20次,那命令应当是:
jstat -gc 9527 250 20
1.2.2 jstat常用选项
-class | 监视类装载、卸载数量、总空间以及类装载所耗费的时间 |
---|---|
-gc | 监视Java堆状况,包括Eden区、两个survivor区、年老代、永久代的容量、已用空间、gc时间合计信息 |
-gccapacity | 监视内容与-gc基本相同,但输出主要关注Java堆各区域使用到的最大、最小空间 |
-gcutil | 监视内容与-gc基本相同,但输出主要关注已使用空间占总空间的百分比 |
-gccause | 与-gcutil功能一样,但会额外输出导致上一次gc产生的原因 |
-compiler | 输出JIT编译器编译过的方法、耗时等信息 |
-printcompilation | 输出已经被JIT编译的方法 |
1.3 jinfo
jinfo(Configuration Info for Java)的作用是实时的查看和调整虚拟机各项参数。
1.3.1 jinfo命令格式
jinfo [option] pid
1.3.2 jinfo常用选项
-flags | 查看jvm的参数 |
---|---|
-sysprops | 查看java系统参数 |
-flag +[参数名 参数值] | 修改jvm参数 |
-v | 查看虚拟机启动时显式指定的参数列表 |
1.4 jmap
jmap(Memory Map for Java)命令用于生成堆转储快照(一般称为heapdump或dump文件)。jmap的作用并不仅仅是为了获取dump文件,它还可以查询finalize执行队列、 Java堆和永久代的详细信息,如空间使用率、 当前用的是哪种收集器等。和jinfo命令一样,jmap有不少功能在Windows平台下都是受限的,除了生成dump文件的-dump选项和用于查看每个类的实例、 空间占用统计的-histo选项在所有操作系统都提供之外,其余选项都只能在Linux/Solaris下使用。
1.4.1 jmap命令格式
jmap [option] vmid
1.4.2 jmap常用选项
-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文件的分析结果后,可以在浏览器中查看。
1.6 jstack
jstack(Stack Trace for Java)命令用于生成虚拟机当前时刻的线程快照(一般称为 threaddump或者javacore文件)。 线程快照就是当前虚拟机内每一条线程正在执行的方法堆栈的集合,生成线程快照的主要目的是定位线程出现长时间停顿的原因,如线程间死锁、死循环、 请求外部资源导致的长时间等待等都是导致线程长时间停顿的常见原因。 线程出现停顿的时候通过jstack来查看各个线程的调用堆栈,就可以知道没有响应的线程到底在后台做些什么事情,或者等待着什么资源。
1.6.1 jstack命令格式
jstack [option] vmid
1.6.2 jstack常用选项
-F | 当正常输出的请求不被响应时,强制输出线程堆栈 |
---|---|
-l | 除堆栈外,显示关于锁的附加信息 |
-m | 如果调用到本地方法的话,可以显示C/C++的堆栈 |
2 可视化工具
2.1 JConsole
JConsole(Java Monitoring and Management Console)是一种基于JMX的可视化监视、 管
理工具。 它管理部分的功能是针对JMX MBean进行管理,由于MBean可以使用代码、 中间件
服务器的管理控制台或者所有符合JMX规范的软件进行访问。
2.2 JVisualVM
VisualVM(All-in-One Java Troubleshooting Tool)是到目前为止随JDK发布的功能最强大的运行监视和故障处理程序,是官方主力发展的虚拟机故障处理工具。 官方在VisualVM的软件说明中写上了“All-in-One”的描述字样,预示着它除了运行监视、 故障处理外,还提供了很多其他方面的功能。 如性能分析(Profiling),VisualVM的性能分析功能甚至比起JProfiler、 YourKit等专业且收费的Profiling工具都不会逊色多少,而且VisualVM的还有一个很大的优点:不需要被监视的程序基于特殊Agent运行,因此它对应用程序的实际性能的影响很小,使得它可以直接应用在生产环境中。 这个优点是JProfiler、 YourKit等工具无法与之媲美的。
2.2.1 生成、 浏览堆转储快照
在VisualVM中生成dump文件有两种方式,可以执行下列任一操作:
在“应用程序”窗口中右键单击应用程序节点,然后选择“堆Dump”。
在“应用程序”窗口中双击应用程序节点以打开应用程序标签,然后在“监视”标签中单击“堆Dump”。
生成了dump文件之后,应用程序页签将在该堆的应用程序下增加一个以[heapdump]开头的子节点,并且在主页签中打开了该转储快照,如果需要把dump文件保存或发送出去,要在heapdump节点上右键选择“另存为”菜单,否则当VisualVM关闭时,生成的dump文件会被当做临时文件删除掉。 要打开一个已经存在的dump文件,通过文件菜单中的“装入”功能,选择硬盘上的dump文件即可。
从堆页签中的“摘要”面板可以看到应用程序dump时的运行时参数、System.getProperties()的内容、 线程堆栈等信息,“类”面板则是以类为统计口径统计类的实例数量、 容量信息,“实例”面板不能直接使用,因为不能确定用户想查看哪个类的实例,所以需要通过“类”面板进入,在“类”中选择一个关心的类后双击鼠标,即可在“实例”里面看见此类中500个实例的具体属性信息。
2.2.2 分析程序性能
在Profiler页签中,VisualVM提供了程序运行期间方法级的CPU执行时间分析以及内存分析,做Profiling分析肯定会对程序运行性能有比较大的影响,所以一般不在生产环境中使用这项功能。
要开始分析,先选择“CPU”和“内存”按钮中的一个,然后切换到应用程序中对程序进行操作,VisualVM会记录到这段时间中应用程序执行过的方法。 如果是CPU分析,将会统计每个方法的执行次数、 执行耗时;如果是内存分析,则会统计每个方法关联的对象数以及这些对象所占的空间。 分析结束后,点击“停止”按钮结束监控过程。
2.2.3 BTrace动态日志跟踪
BTrace是一个很“有趣”的VisualVM插件,本身也是可以独立运行的程序。 它的作用是在不停止目标程序运行的前提下,通过HotSpot虚拟机的HotSwap技术动态加入原本并不存在的调试代码。 这项功能对实际生产中的程序很有意义:经常遇到程序出现问题,但排查错误的一些必要信息,譬如方法参数、 返回值等,在开发时并没有打印到日志之中,以至于不得不停掉服务,通过调试增量来加入日志代码以解决问题。 当遇到生产环境服务无法随便停止时,缺一两句日志导致排错进行不下去是一件非常郁闷的事情。
在VisualVM中安装了BTrace插件后,在应用程序面板中右键点击要调试的程序,会出现“Trace Application……”菜单,点击将进入BTrace面板。 这个面板里面看起来就像一个简单的Java程序开发环境,里面还有一小段Java代码。