常做Java开发的朋友,不管是做纯后端开发,还是做Java Web开发,有一些操作你一定会经常遇到。比如:
列出当前系统上运行了哪些Java应用
这些应用的部署路径是什 么
某个Java应用的堆内存使用情况等
遇到类似的情况,你一般是怎么解决的?
我在工作中见过许多人,想要查看当前正在运行的Java应用时,会通过进程管理器(Windows下),或者使用
ps -ef | grep java (在Linux下使用命令行)。通过这些来获取运行的Java应用列表,以及对应的PID(进程ID)。
而查看堆内存,可能会下载一些第三方的工具等。
其实,JDK中已经默认自带了一些小工具,非常好用,使用方便快捷。不信你看下面这些:
jps (Java Virtual Mathine Process Status Tool)
从名称可以看出个大概。主要是把Java进程的状态列出来。
这个命令可以指定hostid,从而列出远程主机的Java进程。
这个命令有几个参数,我比较常用的有
-l 输出应用程序主类的完整包名,或者是应用程序JAR文件的完整路径。
-v 输出传给JVM的参数。
一般通过jps -lv就可以把本地的所有Java进程,以及传给JVM的参数,运行的文件位置都列出来。这样,一个命令上面前两个问题就解决了。
2. jinfo (Java Configuration Info)
主要用来查看Java进程使用的一些配置参数,比如设置的堆内存有多大,是否允许在OOM产生的时候dump出堆信息等。
除此之外,该命令还可以动态更改配置的项,比如上面提到的OOM dump这个开关是关着的,那可以在运行的时候再打开它。
3. jstate (Java Virtual Machine Statistics Monitoring Tool)
这个命令选项很多,根据不同的选项,可以统计你感兴趣的不同的数据,比如你对JVM的垃圾回收数据感兴趣,可以直接通过指定 -gc 来查看,而更常用的一个选项是 -gcutil 可以统计新生代,老年代垃圾收集的情况。
当然,JDK自带的小工具还有很多,以上面这几个工具为例,其中后两者在执行时,都需要先知道JVM的PID之后,才能执行,毕竟这些工具需要了解你要查看哪个JVM。
下面我们来简要了解下jps的大概实现原理。
每个JVM在启动后,其本质上依然是操作系统的一个进程,都会存在一个PID值与之对应。而JVM内部的实现,则是在每次启动之后,在本地的一个目录内创建一个以当前JVM的PID为文件名的文件,文件路径和操作系统有关,我的windows7下路径是
C:\Users{这里是你的用户名}\AppData\Local\Temp\hsperfdata_{这个也是你的用户名}
在这个目录下,所有你下在运行的JVM对应的PID文件都在这里了。下图是我的列表
同样的道理,如果你曾经用过JConsole(这也是个JDK自带的小工具),可能马上就有感觉了。因为它也是通过读这个目录下的文件信息得到需要的数据,你看下面这个Jconsole新建连接的图,这些PID刚好这上面的文件对应是不是。
PS: 这些原理,我是通过看OpenJDK源码了解到的,如果大家感兴趣,可以一起交流。