一个很强大的免费工具:JDK6自带的jvisualvm。

其实还有个自带工具 jconsole   这里不多说, 用法跟 jvisualvm 差不多。 都是输入命令启动

这个东西我以前听过说,但一直没有用过。看到它提供的截图中可以看到各线程的运行状态,所以准备试一下。这里记录一下使用过程。

1。启动

在命令行输入jvisualvm。如果jdk安装正确的话(6.x以上版本),就会看到如下的一个窗口:

jdk自带检测内存,cpu,线程 的工具——jvisualvm_标签

看起来相当简洁,不像是很强大的样子。

2。运行一个Java程序IncTestN,jvisualvm会自动找到它

jdk自带检测内存,cpu,线程 的工具——jvisualvm_标签_02

3.右键点击它,”打开”

jdk自带检测内存,cpu,线程 的工具——jvisualvm_标签_03

可以看到它有很多标签页,可以让我们监测程序的各种数据。默认没有这么多,我其实安装了一些jvisualvm的插件。

4。查看jvm参数及系统属性

jdk自带检测内存,cpu,线程 的工具——jvisualvm_Java_04

5。查看cpu、内存、类、线程的统计数据

jdk自带检测内存,cpu,线程 的工具——jvisualvm_Java_05

注意,右边第一个还可以查看PermGen。对于scala程序,因为它产生了大量的类定义,所以PermGen有可能会不足,可通过该选项查看PermGen,适当调整:

jdk自带检测内存,cpu,线程 的工具——jvisualvm_记录_06

可以看到,对于本程序来说,PermGen还是比较充分的,无须调整。

6。查看各线程运行情况

这个是重点,我们需要知道各线程的运行情况,特别是否被synchronized阻塞了。

jdk自带检测内存,cpu,线程 的工具——jvisualvm_记录_07

注意右下角,有四个状态说明,分别是:

  1. 运行(Running):我们最喜欢的状态。说明该线程正在执行代码,没有问题。

  2. 休眠(Sleeping):调用了Thread.sleep后的状态,说明线程正停在某个Thread.sleep处

  3. 等待(Wait):手动调用了wait方法,或者某些IO操作,在阻塞中等待数据。

  4. 监视(Monitor):这里就是我想找的问题了。它表示线程想执行一段synchronized中的代码,但是发现已经有其它线程正在执行,自己被block了,只能无奈地等待。如果这种状态多,说明程序需要好好优化。

从上面的这个图可以看到,下面多个线程都处于”监视”状态。多个线程都卡在了独木桥的一头过不去,干不了活干着急呢。

当然这个程序是我专门设计成这样的,存在着严重的性能问题,需要好好优化。

7。查看各线程的统计数据

如果我们需要一些统计数据,比如某个线程总共运行了多少时间,”运行”状态有多久(或百分比),休眠、等待、监视有多久,则需要用到”表”这一页。

jdk自带检测内存,cpu,线程 的工具——jvisualvm_标签_08

从中可以看到这个悲催的程序,几分所有的时间都用在了synchronized的阻塞上了。只有百分之零点几的时间在运行中,效率可真低啊。

还可以使用图表方式来看这些数据,得到更直观的体验:

jdk自带检测内存,cpu,线程 的工具——jvisualvm_记录_09

8。查看各方法的运行时间

想不想程序中到底是哪些方法一直在运行?可使用”抽样器”功能: