자바
[Java] JVM 성능 모니터링을 제공하는 JDK 기본 툴
제리92
2021. 12. 7. 18:01
JDK에서는 개발 및 운영을 위해 기본 툴을 제공합니다.
Java 홈디렉토리의 bin에 툴이 포함되어있습니다.
1.java home path 찾기
root@ureasolution:~# echo $JAVA_HOME
/usr/lib/jvm/java-1.11.0-openjdk-amd64
2.Java 홈디렉토리의 bin 경로로 이동
root@ureasolution:~# cd /usr/lib/jvm/java-1.11.0-openjdk-amd64
root@ureasolution:/usr/lib/jvm/java-1.11.0-openjdk-amd64# cd bin
3.bin 안의 도구들 확인 (JDK 11 기준)
root@ureasolution:/usr/lib/jvm/java-1.11.0-openjdk-amd64/bin# ls -al
total 500
drwxr-xr-x 2 root root 4096 Nov 22 20:30 .
drwxr-xr-x 9 root root 4096 Nov 22 20:30 ..
-rwxr-xr-x 1 root root 10376 Apr 21 2021 jaotc
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jar
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jarsigner
-rwxr-xr-x 1 root root 10304 Apr 21 2021 java
-rwxr-xr-x 1 root root 10352 Apr 21 2021 javac
-rwxr-xr-x 1 root root 10352 Apr 21 2021 javadoc
-rwxr-xr-x 1 root root 10320 Apr 21 2021 javap
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jcmd
-rwxr-xr-x 1 root root 10368 Apr 21 2021 jconsole
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jdb
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jdeprscan
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jdeps
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jfr
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jhsdb
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jimage
-rwxr-xr-x 1 root root 10352 Apr 21 2021 jinfo
-rwxr-xr-x 1 root root 10352 Apr 21 2021 jjs
-rwxr-xr-x 1 root root 10352 Apr 21 2021 jlink
-rwxr-xr-x 1 root root 10352 Apr 21 2021 jmap
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jmod
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jps
-rwxr-xr-x 1 root root 10352 Apr 21 2021 jrunscript
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jshell
-rwxr-xr-x 1 root root 10352 Apr 21 2021 jstack
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jstat
-rwxr-xr-x 1 root root 10320 Apr 21 2021 jstatd
-rwxr-xr-x 1 root root 10320 Apr 21 2021 keytool
-rwxr-xr-x 1 root root 10320 Apr 21 2021 pack200
-rwxr-xr-x 1 root root 10320 Apr 21 2021 rmic
-rwxr-xr-x 1 root root 10320 Apr 21 2021 rmid
-rwxr-xr-x 1 root root 10320 Apr 21 2021 rmiregistry
-rwxr-xr-x 1 root root 10320 Apr 21 2021 serialver
-rwxr-xr-x 1 root root 107408 Apr 21 2021 unpack200
사용 해보기
최근 개발한 요소수소문 서비스를 대상으로 툴을 사용해 보았습니다.
메모리 측정 관련 도구
jps
현재 실행되고 있는 JVM 프로세스 확인
- 28140 pid로 요소수소문 서비스가 구동중인 것을 확인
root@ureasolution:/usr/lib/jvm/java-1.11.0-openjdk-amd64/bin# jps 6483 Jps 28140 ureasolution-0.0.1-main.jar
[옵션]
- q : main 메서드로 전달된 class, jar, 파라미터들을 출력하지 않음 => pid만 출력
root@ureasolution:/usr/lib/jvm/java-1.11.0-openjdk-amd64/bin# jps -q
9001
28140
- m : main 메서드로 전달된 파라미터를 포함하여 출력
root@ureasolution:/usr/lib/jvm/java-1.11.0-openjdk-amd64/bin# jps -m
9024 Jps -m
28140 ureasolution-0.0.1-main.jar --spring.config.name=application-prod
- l : main class 혹은 jar의 full name 출력 => 실행파일의 경로를 포함하여 출력
root@ureasolution:/usr/lib/jvm/java-1.11.0-openjdk-amd64/bin# jps -l
9042 jdk.jcmd/sun.tools.jps.Jps
28140 /[경로]/ureasolution-0.0.1-main.jar
- v : 프로세스 실행시 JVM으로 전달한 파라미터를 포함하여 출력
root@ureasolution:/usr/lib/jvm/java-1.11.0-openjdk-amd64/bin# jps -v
9072 Jps -Dapplication.home=/usr/lib/jvm/java-11-openjdk-amd64 -Xms8m -Djdk.module.main=jdk.jcmd
28140 ureasolution-0.0.1-main.jar
jstack
Thread의 스택 정보를 보기 위해 주로 사용
[옵션]
- l : 추가적인 lock 정보를 포함하여 출력
- m : mixed mode. java stack 정보뿐 아니라 JNI를 통해 실행하는 C/C++ 네이티브 메서드의 stack까지 포함하여 출력
root@ureasolution:/usr/lib/jvm/java-1.11.0-openjdk-amd64/bin# jstack -l 28140
2021-12-07 17:54:19
Full thread dump OpenJDK 64-Bit Server VM (11.0.11+9-Ubuntu-0ubuntu2.18.04 mixed mode, sharing):
Threads class SMR info:
_java_thread_list=0x00007fd1f000e870, length=39, elements=ä
0x00007fd23011f000, 0x00007fd230121000, 0x00007fd230126800, 0x00007fd230128800,
0x00007fd23012a800, 0x00007fd23012d000, 0x00007fd23012f000, 0x00007fd23016a000,
0x00007fd23079c800, 0x00007fd230a13800, 0x00007fd230a32800, 0x00007fd230a34800,
0x00007fd230a44800, 0x00007fd231405800, 0x00007fd2317c0000, 0x00007fd2317c1800,
0x00007fd23135e000, 0x00007fd230016800, 0x00007fd1fc004800, 0x00007fd1fc005800,
0x00007fd1fc007000, 0x00007fd1fc009000, 0x00007fd1fc00b000, 0x00007fd1fc00c000,
0x00007fd1fc00d800, 0x00007fd1fc00f800, 0x00007fd1fc011800, 0x00007fd1fc013000,
0x00007fd20400b800, 0x00007fd20400d000, 0x00007fd20400e000, 0x00007fd20400f800,
0x00007fd204011800, 0x00007fd204013000, 0x00007fd204014000, 0x00007fd204015800,
0x00007fd204017000, 0x00007fd204019000, 0x00007fd1e8025800
å
"Reference Handler" #2 daemon prio=10 os_prio=0 cpu=7.65ms elapsed=422027.01s tid=0x00007fd23011f000 nid=0x6df4 waiting on condition Ä0x00007fd21013e000Å
java.lang.Thread.State: RUNNABLE
at java.lang.ref.Reference.waitForReferencePendingList(java.base@11.0.11/Native Method)
at java.lang.ref.Reference.processPendingReferences(java.base@11.0.11/Reference.java:241)
at java.lang.ref.Reference$ReferenceHandler.run(java.base@11.0.11/Reference.java:213)
Locked ownable synchronizers:
- None
"Finalizer" #3 daemon prio=8 os_prio=0 cpu=0.74ms elapsed=422027.01s tid=0x00007fd230121000 nid=0x6df5 in Object.wait() Ä0x00007fd2037fe000Å
java.lang.Thread.State: WAITING (on object monitor)
at java.lang.Object.wait(java.base@11.0.11/Native Method)
- waiting on <0x00000000c29c1418> (a java.lang.ref.ReferenceQueue$Lock)
at java.lang.ref.ReferenceQueue.remove(java.base@11.0.11/ReferenceQueue.java:155)
- waiting to re-lock in wait() <0x00000000c29c1418> (a java.lang.ref.ReferenceQueue$Lock)
at java.lang.ref.ReferenceQueue.remove(java.base@11.0.11/ReferenceQueue.java:176)
at java.lang.ref.Finalizer$FinalizerThread.run(java.base@11.0.11/Finalizer.java:170)
Locked ownable synchronizers:
- None
"Signal Dispatcher" #4 daemon prio=9 os_prio=0 cpu=0.43ms elapsed=422027.01s tid=0x00007fd230126800 nid=0x6df6 runnable Ä0x0000000000000000Å
java.lang.Thread.State: RUNNABLE
Locked ownable synchronizers:
- None
"Service Thread" #5 daemon prio=9 os_prio=0 cpu=0.15ms elapsed=422027.01s tid=0x00007fd230128800 nid=0x6df7 runnable Ä0x0000000000000000Å
java.lang.Thread.State: RUNNABLE
Locked ownable synchronizers:
- None
jstat
JVM 성능을 통계내는 툴 입니다.
[옵션]
- class
클래스 로더 수행내역 분석 - compiler
HotSpot(JIT) 컴파일러 수행내역 분석 - gc
heap에서 gc를 수행한 내역 분석 - printcompilation
HotSpot 컴파일러가 컴파일한 메서드 내역 분석
jmap
현재 실행중인 프로세스의 JVM 메모리 맵을 보여주는 분석 도구
[옵션]
- jmap -clstats
- 프로세스의 클래스 로더를 분석
root@ureasolution:/usr/lib/jvm/java-1.11.0-openjdk-amd64/bin# jmap -clstats 28140
Index Super InstBytes KlassBytes annotations CpAll MethodCount Bytecodes MethodAll ROAll RWAll Total ClassName
1 -1 5997536 504 0 0 0 0 0 24 616 640 ÄB
2 11 1771320 624 128 14272 109 4576 71248 20576 67440 88016 java.lang.String
3 11 1675456 680 0 22120 139 5679 87664 28040 84896 112936 java.lang.Class
4 11 1630784 600 0 1368 9 213 2880 1632 3592 5224 java.util.concurrent.ConcurrentHashMap$Node
5 -1 1552848 504 0 0 0 0 0 24 616 640 ÄC
6 -1 1046088 504 0 0 0 0 0 24 616 640 ÄI
7 6837 987184 1200 0 6824 47 1117 27936 7024 29872 36896 java.lang.reflect.Method
8 -1 734728 504 0 0 0 0 0 24 616 640 ÄLjava.lang.Object;
9 13 514640 592 0 512 1 10 624 312 1656 1968 java.util.LinkedHashMap$Entry
10 -1 486056 504 0 0 0 0 0 32 616 648 ÄLjava.util.concurrent.ConcurrentHashMap$Node;
11 -1 474928 528 0 1280 14 109 4704 1624 5184 6808 java.lang.Object
12 -1 450952 504 0 0 0 0 0 32 616 648 ÄLjava.util.HashMap$Node;
13 11 432064 592 0 1392 7 149 2296 1200 3432 4632 java.util.HashMap$Node
14 25 315504 1136 0 3816 26 879 14360 5048 14984 20032 java.util.LinkedHashMap
15 -1 212496 504 0 0 0 0 0 56 616 672 ÄLjava.lang.Class;
16 7422 131640 872 0 2992 10 259 4464 1768 6920 8688 org.springframework.boot.loader.jar.JarEntry
17 11 126288 584 0 11528 89 4308 56216 16648 53400 70048 java.lang.invoke.MemberName
18 6829 117520 568 0 696 3 56 1888 504 2848 3352 java.lang.ref.SoftReference
19 11 109584 552 72 1512 6 242 4704 1296 5792 7088 org.springframework.core.MethodClassKey
20 7185 96144 1448 0 7008 64 2677 39552 12392 36872 49264
- jmap -finalizerinfo
- 프로세스의 JVM 힙에서 finalization 대기중인 객체 정보 출력
- jmap -histo[:live]
- 프로세스의 JVM 힙에 있는 객체 정보를 히스토그램화 하여 출력(live 옵션이 붙으면 살아있는 객체만을 대상으로함)
- 가장 많이 메모리를 점유한 객체부터 데이터를 출력
- 특정 객체가 비정상적으로 많이 생성되어있고 숫자가 지속적으로 늘어난다면 메모리 누수를 유추해볼 수 있음
root@ureasolution:/usr/lib/jvm/java-1.11.0-openjdk-amd64/bin# jmap -histo:live 28140
num #instances #bytes class name (module)
-------------------------------------------------------
1: 82066 5988008 [B (java.base@11.0.11)
2: 73846 1772304 java.lang.String (java.base@11.0.11)
3: 14129 1674000 java.lang.Class (java.base@11.0.11)
4: 51683 1653856 java.util.concurrent.ConcurrentHashMap$Node (java.base@11.0.11)
5: 4836 1496072 [C (java.base@11.0.11)
6: 6993 1045272 [I (java.base@11.0.11)
7: 10878 957264 java.lang.reflect.Method (java.base@11.0.11)
8: 11468 732896 [Ljava.lang.Object; (java.base@11.0.11)
9: 402 485704 [Ljava.util.concurrent.ConcurrentHashMap$Node; (java.base@11.0.11)
10: 29681 474896 java.lang.Object (java.base@11.0.11)
...
- jmap -dump:
- 프로세스의 JVM heap에 대해 덤프를 생성
root@ureasolution:/usr/lib/jvm/java-1.11.0-openjdk-amd64/bin# jmap -dump:format=b,file=heapdump.hprof 28140
- 생성된 dump 파일은 외부 툴을 이용해서 분석해볼 수 있습니다.
- jdk에 포함되어있던 jhat으로 분석할 수 있었는데 jdk 9 버전부터 jhat이 제외되었습니다.
- (JEP 241 - jhat is an experimental, unsupported, and out-of-date tool.)
- jhat은 jdk 6에 포함되었던 툴인데, 현재는 우수한 다른 툴들이 많이 나와서 거의 사용하지 않는 툴이 되었기에 제외되었다고 합니다.
VisualVM
- dump 파일 분석 툴
- jdk 기본 툴은 아니지만 생성한 dump 파일을 분석하기에 좋은 툴입니다.
[참고]