一、介绍

JVisualVM,能够监控线程,内存情况,查看方法的CPU时间和内存中的对 象,已被GC的对象,反向查看分配的堆栈(如100个String对象分别由哪几个对象分配出来的).

从界面上看还是比较简洁的,左边是树形结构,自动显示当前本机所运行的Java程序,还可以添加远程的Java VM,其中括号里面的PID指的是进程ID。OverView界面显示VM启动参数以及该VM对应的一些属性。Monitor界面则是监控Java堆大小,Permgen大小,Classes和线程数量。jdk不同版本中界面会不太一致,如有的含cpu监控,有的则不含(jdk1.6.0_10未包含)。

二、JVisualVM能做什么

VisualVM 是Netbeans的profile子项目,已在JDK6.0 update 7 中自带,能够监控线程,内存情况,查看方法的CPU时间和内存中的对 象,已被GC的对象,反向查看分配的堆栈(如100个String对象分别由哪几个对象分配出来的)。

三、使用

1、监控本地的java进程

jvisualvm位于JAVA_HOME/bin目录下,如下图:

java visualvm怎么连接远程地址 java visualvm远程监控_Java


直接双击就可以打开该程序,打开后界面如下:

java visualvm怎么连接远程地址 java visualvm远程监控_java_02


如果只是监控本地的java进程,是不需要配置参数的,直接打开就能够进行监控。首先我们需要在本地打开一个Java程序,例如我打开IDEA,这时在jvisualvm界面就可以看到与IDEA相关的Java进程了:

java visualvm怎么连接远程地址 java visualvm远程监控_Java_03


注意:在使用jonsole、jvisualvm时,不显示本地Java进程信息点击一个进程,就可以看到该进程的概述信息,该进程的JVM参数以及系统属性等信息都能够查看到:

java visualvm怎么连接远程地址 java visualvm远程监控_Java_04


点击 “监视” 就能够看到CPU、内存、类以及线程的活动状况,点击右上角的 “堆Dump” 就能够导出内存映像文件:

java visualvm怎么连接远程地址 java visualvm远程监控_JAVA_05


除了导出,也还可以导入内存映像文件,只不过分析功能上没有MAT强大:

java visualvm怎么连接远程地址 java visualvm远程监控_java_06


点击 “线程” 就能够看到该进程内部的所有线程,以及线程的运行状况等信息。如果点击右上角的 “线程Dump” 就会导出一个内容与jstack打印内容一致的文件:

运行:正在运行的
休眠:sleep
等待:wait
驻留:线程池里面空闲的线程
监视:阻塞的线程,正在等待锁

java visualvm怎么连接远程地址 java visualvm远程监控_java_07


点击 “抽样器” 界面中的 “CPU ” 就可以动态的看到每个方法的执行时间,当我们的代码执行的比较慢了,就可以通过抽样器来查看是哪一个方法执行的比较慢:

java visualvm怎么连接远程地址 java visualvm远程监控_Java_08


而点击 “内存” 的话,就可以实时的、动态的查看到每个类实例对象的数量以及这些实例所占用的内存大小:

java visualvm怎么连接远程地址 java visualvm远程监控_JAVA_09


在Profiler界面上,可以对CPU、内存进行性能分析,分析后会给出分析结果:

java visualvm怎么连接远程地址 java visualvm远程监控_JAVA_10

接下来我们下载两个比较实用的插件,首先到VisualVM插件中心,地址如下:
https://visualvm.github.io/pluginscenters.html.

因为有时候自带的插件地址可能无法使用,所以我们需要自行配置可用的地址:

java visualvm怎么连接远程地址 java visualvm远程监控_Java_11


回到jvisualvm中,点击菜单栏中的 “工具“ 选项,再点击子菜单的 ”插件“ 选项:

java visualvm怎么连接远程地址 java visualvm远程监控_JAVA_12


然后按照下图配置插件地址:

java visualvm怎么连接远程地址 java visualvm远程监控_Java_13


配置好插件地址后,按照下图勾选以下两个插件:

java visualvm怎么连接远程地址 java visualvm远程监控_java_14


按照提示一步步的进行安装:

java visualvm怎么连接远程地址 java visualvm远程监控_Java_15


java visualvm怎么连接远程地址 java visualvm远程监控_java_16


安装完成:

java visualvm怎么连接远程地址 java visualvm远程监控_java_17


插件安装完成后需要重启jvisualvm才会生效,这时会看到选项卡中多了一个 “Visual GC” 选项,点击该选项后,可以动态的查看到JVM内存结构各个区域的运行状况,极大方便我们进行各个区内存的监控及调优:

java visualvm怎么连接远程地址 java visualvm远程监控_JAVA_18


另一个插件的作用就是可以让我们针对某个进程编写BTrace脚本,右键点击一个进程,在弹出来的菜单中,选择“Trace application…”,就可以编写BTrace脚本了:

java visualvm怎么连接远程地址 java visualvm远程监控_Java_19

2、监控远程的java进程

在上一小节中,我们简单介绍了如何使用JDK自带的jvisualvm工具来监控本地的Java进程。而本小节我们将介绍一下如何使用jvisualvm来监控远程的java进程,我们这里以线上服务器的Tomcat为例。

打开jvisualvm,双击 “远程” 选项,添加一个远程主机,即远程的服务器:

java visualvm怎么连接远程地址 java visualvm远程监控_JAVA_20

然后编辑catalina.sh文件,在该文件中加入如下配置内容:
```c
[root@VM-0-5-centos ~]# vim /home/tomcat/apache-tomcat-9.5.8/bin/catalina.sh
JAVA_OPTS="$JAVA_OPTS -Dcom.sun.management.jmxremote -Dcom.sun.management.jmxremote.port=9004 -Dcom.sun.management.jmxremote.authenticate=false -Dcom.sun.management.jmxremote.ssl=false -Djava.net.preferIPv4Stack=true -Djava.rmi.server.hostname=106.52.186.223"

配置简单说明:

  • -Dcom.sun.management.jmxremote添加一个jmx远程连接属性
  • -Dcom.sun.management.jmxremote.port=9004 指定连接的端口号
  • -Dcom.sun.management.jmxremote.authenticate=false 是否启用验证
  • -Dcom.sun.management.jmxremote.ssl=false 是否启用ssl
  • -Djava.net.preferIPv4Stack=true 是否优先使用ipv4
  • -Djava.rmi.server.hostname=106.52.186.223 指定远程主机的ip地址

增加完配置后,重启Tomcat,查看是否有9004端口,有的话就代表配置成功了,如下:

[root@VM-0-5-centos ~]# netstat -lntp | grep java
tcp        0      0 0.0.0.0:37538           0.0.0.0:*               LISTEN      16308/java          
tcp        0      0 0.0.0.0:45033           0.0.0.0:*               LISTEN      16308/java          
tcp        0      0 0.0.0.0:9004            0.0.0.0:*               LISTEN      16308/java          
tcp        0      0 0.0.0.0:8060            0.0.0.0:*               LISTEN      16308/java          
tcp6       0      0 :::40811                :::*                    LISTEN      25513/java          
tcp6       0      0 :::1099                 :::*                    LISTEN      25513/java          
tcp6       0      0 :::8050                 :::*                    LISTEN      25513/java

回到jvisualvm中,添加JMX连接,配置连接地址和端口,如下:

java visualvm怎么连接远程地址 java visualvm远程监控_java_21


java visualvm怎么连接远程地址 java visualvm远程监控_JAVA_22


注意:如果你使用的是阿里云等云服务器的话,不仅需要配置防火墙规则来开放端口,还需要到云服务器的安全组规则中,增加相应的端口规则。连接成功后,如下:

java visualvm怎么连接远程地址 java visualvm远程监控_java_23


同样的,可以像监控本地进程一样,监控远程的进程,在界面的操作上是一模一样的。唯一不同的也就是需要添加一个远程主机,然后到远程的Java进程上配置一些jmx参数而已:

java visualvm怎么连接远程地址 java visualvm远程监控_java_24


以上我们是以Tomcat这种服务型的进程作为一个示例,只需要去配置Tomcat的脚本文件就可以了。那么如果我们需要监控的是自己线上跑的一个Java进程需要怎么进行配置呢?

方式一

使用这些参数,只不过是作为启动参数,在启动项目的时候加上即可,如下:

[root@VM-0-5-centos ~]#  nohup java -Dcom.sun.management.jmxremote -Dcom.sun.management.jmxremote.port=9004 -Dcom.sun.management.jmxremote.authenticate=false -Dcom.sun.management.jmxremote.ssl=false -Djava.net.preferIPv4Stack=true -Djava.rmi.server.hostname=106.52.186.223 -jar boke-2.0.jar  > /data/mylog.log 2>&1

方式二

编辑/etc/profile文件,在该文件中加入如下配置内容:

[root@VM-0-5-centos ~]# vi /etc/profile
export JAVA_OPTS="-Dcom.sun.management.jmxremote -Dcom.sun.management.jmxremote.port=9004 -Dcom.sun.management.jmxremote.authenticate=false -Dcom.sun.management.jmxremote.ssl=false -Djava.net.preferIPv4Stack=true -Djava.rmi.server.hostname=106.52.186.223"

java visualvm怎么连接远程地址 java visualvm远程监控_JAVA_25


nohup java $JAVA_OPTS -jar boke-2.0.jar > /data/mylog.log 2>&1成功启动后,也是使用同样的方式进行连接,注意端口不要弄错了:

java visualvm怎么连接远程地址 java visualvm远程监控_JAVA_22


连接成功后也是一样的:

java visualvm怎么连接远程地址 java visualvm远程监控_java_23


然后我们就可以愉快的在本地监控远程的Java进程了,而且还是图形化的,免去了敲命令的烦恼。

所以线上监控不能少,是我们必须要掌握的一种技能。监控相当于我们的眼睛,如果没有监控工具,那我们就相当于瞎子一样两眼一抹黑。无法看到内存、线程的使用情况,当出现异常的时候,也难以定位问题发生的原因。