JVM诊断命令jcmd介绍

原创:扣钉日记(微信公众号ID:codelogs),欢迎分享,转载请保留出处 。
简介从JDK7开始,jdk提供了一个方便扩展的诊断命令jcmd,用来取代之前比较分散的jdk基础命令,如jps、jstack、jmap、jinfo等,并且jdk添加新的诊断功能,也会通过jcmd提供,所以还是有必要将这个命令熟悉起来的 。
列出JAVA进程jps提供了列出本机java进程的功能,jcmd与之类似,直接输入jcmd即可,如下:
$ jcmd10732 App.jar10767 sun.tools.jcmd.JCmd可以看到,本机有一个app.jar的java进程 。
列出jcmd支持的子命令$ jcmd 10732 help10732:The following commands are available:VM.native_memoryVM.classloader_statsThread.printGC.class_statsGC.class_histogramGC.heap_dumpGC.finalizer_infoGC.heap_infoGC.run_finalizationGC.runVM.uptimeVM.dynlibsVM.flagsVM.system_propertiesVM.command_lineVM.versionhelp【JVM诊断命令jcmd介绍】如上,可以看到,10732这个java进程支持了不少子命令呢!
查看java线程栈jcmd可以像jstack一样,打印java线程栈,使用jcmd 0 Thread.print即可,如下:
JVM诊断命令jcmd介绍

文章插图
 
注:jcmd有个默认行为,当传递给jcmd的进程id是0时,jcmd会在本机所有java进程中执行子命令,这样我们就可以少操作一步了 。
查看jvm堆情况jcmd可以快速查看当前堆容量、已使用容量等等关键信息,如下:
$ jcmd 0 GC.heap_info10732: garbage-first heaptotal 204800K, used 44778K [0x00000006f9a00000, 0x00000006f9b00640, 0x00000007c0000000)region size 1024K, 44 young (45056K), 5 survivors (5120K) Metaspaceused 16876K, capacity 18012K, committed 18304K, reserved 1064960Kclass spaceused 2101K, capacity 2330K, committed 2432K, reserved 1048576K也可以像jmap一样查看堆直方图并转储堆内存到文件中,以分析内存不合理的占用问题,如下:
# 堆直方图,查看哪些类的对象实例最多$ jcmd 0 GC.class_histogram 10732: num#instances#bytesclass name----------------------------------------------1:126591142376[C2:3649403376java.lang.Class3:12644303456java.lang.String4:9020288640java.util.concurrent.ConcurrentHashMap$Node5:1378280376[B6:2241183608[I7:3351164296[Ljava.lang.Object;8:1554133496[Ljava.util.HashMap$Node;9:1192104896java.lang.reflect.Method10:77104016[Ljava.util.concurrent.ConcurrentHashMap$Node;11:6307100912java.lang.Object12:2517100680java.util.LinkedHashMap$Entry13:307198272java.util.HashMap$Node14:114855104java.util.HashMap15:196246856[Ljava.lang.Class;16:78243792java.util.LinkedHashMap# 导出堆内存文件$ jcmd 0 GC.heap_dump /home/work/heap.hprof11377:Heap dump file created查看jvm属性等使用jinfo可以查看jvm属性与参数,jcmd同样也可以做到,如下:
# 查看jvm参数$ jcmd 0 VM.flags11377:-XX:CICompilerCount=4 -XX:ConcGCThreads=2 -XX:G1HeapRegionSize=1048576 -XX:InitialHeapSize=209715200 -XX:MarkStackSize=4194304 -XX:MaxHeapSize=3328180224 -XX:MaxNewSize=1996488704 -XX:MinHeapDeltaBytes=1048576 -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseG1GC# 查看jvm属性$ jcmd 0 VM.system_properties11377:#Sat Jul 30 16:50:33 CST 2022java.runtime.name=OpenJDK Runtime Environmentjava.protocol.handler.pkgs=org.springframework.boot.loadersun.boot.library.path=/opt/jdk8u212-b03/jre/lib/amd64java.vm.version=25.212-b03java.vm.vendor=AdoptOpenJDKjava.vendor.url=http://java.oracle.com/path.separator=:java.vm.name=OpenJDK 64-Bit Server VM查看堆外内存分配情况为了提升性能,很多基础框架使用了堆外内存,而jcmd可以检查堆外内存的使用情况,如下:
$ jcmd 0 VM.native_memory11817:Native Memory Tracking:Total: reserved=4813774KB, committed=357358KB-Java Heap (reserved=3250176KB, committed=204800KB)(mmap: reserved=3250176KB, committed=204800KB)-Class (reserved=1065417KB, committed=17225KB)(classes #3070)(malloc=457KB #4487)(mmap: reserved=1064960KB, committed=16768KB)-Thread (reserved=33955KB, committed=33955KB)(thread #34)(stack: reserved=33816KB, committed=33816KB)(malloc=100KB #177)(arena=39KB #62)-Code (reserved=250612KB, committed=7124KB)(malloc=1012KB #2197)(mmap: reserved=249600KB, committed=6112KB)-...-Symbol (reserved=4471KB, committed=4471KB)(malloc=3376KB #25096)(arena=1095KB #1)-Arena Chunk (reserved=23038KB, committed=23038KB)(malloc=23038KB)-Unknown (reserved=6348KB, committed=0KB)(mmap: reserved=6348KB, committed=0KB)如上,能够看到jvm原生内存各个段的分配情况 。
注:需要添加jvm参数-XX:NativeMemoryTracking=summary才能使用上述功能 。
jvm性能数据通过PerfCounter.print可以查看jvm运行时的一些性能数据,如下:


推荐阅读