17. JVM之性能调优工具

jmap

命令介绍

得到运行java程序的内存分配的详细情况。例如实例个数,大小等

格式

jmap [ option ] pid
  • pid为java进程号
  • -dump:[live,]format=b,file=<filename> 使用hprof二进制形式,输出jvm的heap内容到文件=. live子选项是可选的,假如指定live选项,那么只输出活的对象到文件.
  • -finalizerinfo 打印正等候回收的对象的信息.
  • -heap 打印heap的概要信息,GC使用的算法,heap的配置及wise heap的使用情况.
  • -histo[:live] 打印每个class的实例数目,内存占用,类全名信息. VM的内部类名字开头会加上前缀“*”. 如果live子参数加上后,只统计活的对象数量.
  • -permstat 打印classload和jvm heap长久层的信息. 包含每个classloader的名字,活泼性,地址,父classloader和加载的class数量. 另外,内部String的数量和占用内存数也会打印出来.
  • -F 强迫.在pid没有相应的时候使用-dump或者-histo参数. 在这个模式下,live子参数无效.
  • -h | -help 打印辅助信息
  • -J 传递参数给jmap启动的jvm.

实例

  • jmap -histo 15042
 num     #instances         #bytes  class name
----------------------------------------------
   1:           585        5025912  [I
   2:          1117         926448  [B
   3:          3424         462928  [C
   4:           621          70760  java.lang.Class
   5:          2372          56928  java.lang.String
   6:           655          35248  [Ljava.lang.Object;
   7:           152          10944  java.lang.reflect.Field
   8:           287           6888  java.lang.StringBuilder
   9:           212           6784  java.io.File
  10:            95           6080  java.net.URL
  • jmap -dump:format=b,file=/Users/liukai/heap.bin 15093
    • 导出内存,据说对性能有影响,小心使用
    • format=b是通过二进制的意思
    • 把内存结构全部dump到二进制文件中,通过IBM的HeapAnalyzer和eclipse的MemoryAnalyzer都可以分析内存结构

jstack

命令介绍

jstack能得到运行java程序的java stack和native stack的信息。可以轻松得知当前线程的运行情况

格式

jstack [ option ] pid
  • pid为jdk进程号
  • 参数 -l 选项用于打印锁的额外信息。

thread dump状态

  • 死锁,Deadlock(重点关注)
  • 等待资源,Waiting on condition(重点关注)
  • 等待获取监视器,Waiting on monitor entry(重点关注)
  • 阻塞,Blocked(重点关注)
  • 执行中,Runnable
  • 暂停,Suspended
  • 对象等待中,Object.wait() 或 TIMED_WAITING
  • 停止,Parked

thread dump中的特殊线程

线程名称 所属 解释说明
Attach Listener JVM Attach Listener 线程是负责接收到外部的命令,而对该命令进行执行并把结果返回给发送者。通常我们会用一些命令去要求jvm给我们一些反馈信息,如:java -version、jmap、jstack等等。 如果该线程在jvm启动的时候没有初始化,那么,则会在用户第一次执行jvm命令时,得到启动。
Signal Dispatcher JVM 前面我们提到第一个Attach Listener线程的职责是接收外部jvm命令,当命令接收成功后,会交给signal dispather 线程去进行分发到各个不同的模块处理命令,并且返回处理结果。 signal dispather线程也是在第一次接收外部jvm命令时,进行初始化工作。
CompilerThread0 JVM 用来调用JITing,实时编译装卸class 。 通常,jvm会启动多个线程来处理这部分工作,线程名称后面的数字也会累加,例如:CompilerThread1

jstack打印信息

参考附1

Jstat

命令介绍

这是一个比较实用的一个命令,主要利用JVM内建的指令对Java应用程序的资源和性能进行实时的命令行的监控,包括了对Heap size和垃圾回收状况,类加载及编译情况的监控。

命令格式

  • -class:统计class loader行为信息
  • -compiler:统计编译行为信息
  • -gc:统计jdk gc时heap信息
  • -gccapacity:统计不同的generations(不知道怎么翻译好,包括新生区,老年区,permanent区)相应的heap容量情况
  • -gccause:统计gc的情况,(同-gcutil)和引起gc的事件
  • -gcnew:统计gc时,新生代的情况
  • -gcnewcapacity:统计gc时,新生代heap容量
  • -gcold:统计gc时,老年区的情况
  • -gcoldcapacity:统计gc时,老年区heap容量
  • -gcpermcapacity:统计gc时,permanent区heap容量
  • -gcutil:统计gc时,heap情况

可通过jstat -options查看支持的参数

输出参数内容

  • S0 — Heap上的 Survivor space 0 区已使用空间的百分比
    • S0C:S0当前容量的大小
    • S0U:S0已经使用的大小
  • S1 — Heap上的 Survivor space 1 区已使用空间的百分比
    • S1C:S1当前容量的大小
    • S1U:S1已经使用的大小
  • E — Heap上的 Eden space 区已使用空间的百分比
    • EC:Eden space当前容量的大小
    • EU:Eden space已经使用的大小
  • O — Heap上的 Old space 区已使用空间的百分比
    • OC:Old space当前容量的大小
    • OU:Old space已经使用的大小
  • P — Perm space 区已使用空间的百分比
    • OC:Perm space当前容量的大小
    • OU:Perm space已经使用的大小
  • YGC — 从应用程序启动到采样时发生 Young GC 的次数
  • YGCT– 从应用程序启动到采样时 Young GC 所用的时间(单位秒)
  • FGC — 从应用程序启动到采样时发生 Full GC 的次数
  • FGCT– 从应用程序启动到采样时 Full GC 所用的时间(单位秒)
  • GCT — 从应用程序启动到采样时用于垃圾回收的总时间(单位秒),它的值等于YGC+FGC

实例

  • jstat -gc 15158
 S0C    S1C    S0U    S1U      EC       EU        OC         OU       MC     MU    CCSC   CCSU   YGC     YGCT    FGC    FGCT     GCT
10752.0 10752.0  0.0    0.0   65536.0   5243.0   175104.0     0.0     4480.0 774.5  384.0   75.9       0    0.000   0      0.000    0.000
  • jstat -compiler 15244
Compiled Failed Invalid   Time   FailedType FailedMethod
      34      0       0     0.01          0
  • jstat -class 15271
Loaded  Bytes  Unloaded  Bytes     Time
   539  1087.0        0     0.0       0.10

附1

bogon:~ liukai$ jstack 14720
2019-06-13 16:42:57
Full thread dump Java HotSpot(TM) 64-Bit Server VM (25.144-b01 mixed mode):

"Attach Listener" #11 daemon prio=9 os_prio=31 tid=0x00007f86a8002000 nid=0x1107 waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"Service Thread" #10 daemon prio=9 os_prio=31 tid=0x00007f86a7068800 nid=0x5503 runnable [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"C1 CompilerThread3" #9 daemon prio=9 os_prio=31 tid=0x00007f86a4874000 nid=0x5303 waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"C2 CompilerThread2" #8 daemon prio=9 os_prio=31 tid=0x00007f86a4873000 nid=0x5103 waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"C2 CompilerThread1" #7 daemon prio=9 os_prio=31 tid=0x00007f86a4872800 nid=0x4f03 waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"C2 CompilerThread0" #6 daemon prio=9 os_prio=31 tid=0x00007f86a4880800 nid=0x4d03 waiting on condition [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"Monitor Ctrl-Break" #5 daemon prio=5 os_prio=31 tid=0x00007f86a4876800 nid=0x4b03 runnable [0x000070000fe62000]
   java.lang.Thread.State: RUNNABLE
	at java.net.SocketInputStream.socketRead0(Native Method)
	at java.net.SocketInputStream.socketRead(SocketInputStream.java:116)
	at java.net.SocketInputStream.read(SocketInputStream.java:171)
	at java.net.SocketInputStream.read(SocketInputStream.java:141)
	at sun.nio.cs.StreamDecoder.readBytes(StreamDecoder.java:284)
	at sun.nio.cs.StreamDecoder.implRead(StreamDecoder.java:326)
	at sun.nio.cs.StreamDecoder.read(StreamDecoder.java:178)
	- locked <0x000000076ac835a8> (a java.io.InputStreamReader)
	at java.io.InputStreamReader.read(InputStreamReader.java:184)
	at java.io.BufferedReader.fill(BufferedReader.java:161)
	at java.io.BufferedReader.readLine(BufferedReader.java:324)
	- locked <0x000000076ac835a8> (a java.io.InputStreamReader)
	at java.io.BufferedReader.readLine(BufferedReader.java:389)
	at com.intellij.rt.execution.application.AppMainV2$1.run(AppMainV2.java:63)

"Signal Dispatcher" #4 daemon prio=9 os_prio=31 tid=0x00007f86a6821800 nid=0x4903 runnable [0x0000000000000000]
   java.lang.Thread.State: RUNNABLE

"Finalizer" #3 daemon prio=8 os_prio=31 tid=0x00007f86a4811800 nid=0x3803 in Object.wait() [0x000070000fbd9000]
   java.lang.Thread.State: WAITING (on object monitor)
	at java.lang.Object.wait(Native Method)
	- waiting on <0x000000076ab08ec8> (a java.lang.ref.ReferenceQueue$Lock)
	at java.lang.ref.ReferenceQueue.remove(ReferenceQueue.java:143)
	- locked <0x000000076ab08ec8> (a java.lang.ref.ReferenceQueue$Lock)
	at java.lang.ref.ReferenceQueue.remove(ReferenceQueue.java:164)
	at java.lang.ref.Finalizer$FinalizerThread.run(Finalizer.java:209)

"Reference Handler" #2 daemon prio=10 os_prio=31 tid=0x00007f86a5008800 nid=0x3603 in Object.wait() [0x000070000fad6000]
   java.lang.Thread.State: WAITING (on object monitor)
	at java.lang.Object.wait(Native Method)
	- waiting on <0x000000076ab06b68> (a java.lang.ref.Reference$Lock)
	at java.lang.Object.wait(Object.java:502)
	at java.lang.ref.Reference.tryHandlePending(Reference.java:191)
	- locked <0x000000076ab06b68> (a java.lang.ref.Reference$Lock)
	at java.lang.ref.Reference$ReferenceHandler.run(Reference.java:153)

"main" #1 prio=5 os_prio=31 tid=0x00007f86a6805000 nid=0x1b03 waiting on condition [0x000070000f0b8000]
   java.lang.Thread.State: TIMED_WAITING (sleeping)
	at java.lang.Thread.sleep(Native Method)
	at com.kevin.Testt.main(Testt.java:39)

"VM Thread" os_prio=31 tid=0x00007f86a5004000 nid=0x3403 runnable

"GC task thread#0 (ParallelGC)" os_prio=31 tid=0x00007f86a680e000 nid=0x2403 runnable

"GC task thread#1 (ParallelGC)" os_prio=31 tid=0x00007f86a7800000 nid=0x2603 runnable

"GC task thread#2 (ParallelGC)" os_prio=31 tid=0x00007f86a7801000 nid=0x2803 runnable

"GC task thread#3 (ParallelGC)" os_prio=31 tid=0x00007f86a7801800 nid=0x2a03 runnable

"GC task thread#4 (ParallelGC)" os_prio=31 tid=0x00007f86a7802000 nid=0x2c03 runnable

"GC task thread#5 (ParallelGC)" os_prio=31 tid=0x00007f86a7802800 nid=0x2e03 runnable

"GC task thread#6 (ParallelGC)" os_prio=31 tid=0x00007f86a7803800 nid=0x3003 runnable

"GC task thread#7 (ParallelGC)" os_prio=31 tid=0x00007f86a7804000 nid=0x3203 runnable

"VM Periodic Task Thread" os_prio=31 tid=0x00007f86a7071000 nid=0x5703 waiting on condition