偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

JVM性能調(diào)優(yōu)監(jiān)控工具使用詳解

云計(jì)算 虛擬化
這些問(wèn)題在日常開(kāi)發(fā)、維護(hù)中可能被很多人忽視(比如有的人遇到上面的問(wèn)題只是重啟服務(wù)器或者調(diào)大內(nèi)存,而不會(huì)深究問(wèn)題根源),但能夠理解并解決這些問(wèn)題是Java程序員進(jìn)階的必備要求。

 [[280944]]

現(xiàn)實(shí)企業(yè)級(jí)Java應(yīng)用開(kāi)發(fā)、維護(hù)中,有時(shí)候我們會(huì)碰到下面這些問(wèn)題:

  • OutOfMemoryError,內(nèi)存不足
  • 內(nèi)存泄露
  • 線(xiàn)程死鎖
  • 鎖爭(zhēng)用(Lock Contention)
  • Java進(jìn)程消耗CPU過(guò)高
  • ......

這些問(wèn)題在日常開(kāi)發(fā)、維護(hù)中可能被很多人忽視(比如有的人遇到上面的問(wèn)題只是重啟服務(wù)器或者調(diào)大內(nèi)存,而不會(huì)深究問(wèn)題根源),但能夠理解并解決這些問(wèn)題是Java程序員進(jìn)階的必備要求。本文將對(duì)一些常用的JVM性能調(diào)優(yōu)監(jiān)控工具進(jìn)行介紹,希望能起拋磚引玉之用。

而且這些監(jiān)控、調(diào)優(yōu)工具的使用,無(wú)論你是運(yùn)維、開(kāi)發(fā)、測(cè)試,都是必須掌握的。

A、 jps(Java Virtual Machine Process Status Tool)

jps主要用來(lái)輸出JVM中運(yùn)行的進(jìn)程狀態(tài)信息。語(yǔ)法格式如下:

  1. jps [options] [hostid] 

如果不指定hostid就默認(rèn)為當(dāng)前主機(jī)或服務(wù)器。

命令行參數(shù)選項(xiàng)說(shuō)明如下:

  1. -q 不輸出類(lèi)名、Jar名和傳入main方法的參數(shù) 
  2. -m 輸出傳入main方法的參數(shù) 
  3. -l 輸出main類(lèi)或Jar的全限名 
  4. -v 輸出傳入JVM的參數(shù) 

比如下面:

  1. root@ubuntu:/# jps -m -l 
  2. 2458 org.artifactory.standalone.main.Main /usr/local/artifactory-2.2.5/etc/jetty.xml 
  3. 29920 com.sun.tools.hat.Main -port 9998 /tmp/dump.dat 
  4. 3149 org.apache.catalina.startup.Bootstrap start 
  5. 30972 sun.tools.jps.Jps -m -l 
  6. 8247 org.apache.catalina.startup.Bootstrap start 
  7. 25687 com.sun.tools.hat.Main -port 9999 dump.dat 
  8. 21711 mrf-center.jar 

B、jstack

jstack主要用來(lái)查看某個(gè)Java進(jìn)程內(nèi)的線(xiàn)程堆棧信息。語(yǔ)法格式如下:

  1. jstack [option] pid 
  2. jstack [option] executable core 
  3. jstack [option] [server-id@]remote-hostname-or-ip 

命令行參數(shù)選項(xiàng)說(shuō)明如下:

  1. -l long listings,會(huì)打印出額外的鎖信息,在發(fā)生死鎖時(shí)可以用jstack -l pid來(lái)觀察鎖持有情況-m mixed mode,不僅會(huì)輸出Java堆棧信息,還會(huì)輸出C/C++堆棧信息(比如Native方法) 

jstack可以定位到線(xiàn)程堆棧,根據(jù)堆棧信息我們可以定位到具體代碼,所以它在JVM性能調(diào)優(yōu)中使用得非常多。下面我們來(lái)一個(gè)實(shí)例找出某個(gè)Java進(jìn)程中最耗費(fèi)CPU的Java線(xiàn)程并定位堆棧信息,用到的命令有ps、top、printf、jstack、grep。

第一步先找出Java進(jìn)程ID,我部署在服務(wù)器上的Java應(yīng)用名稱(chēng)為mrf-center:

  1. root@ubuntu:/# ps -ef | grep mrf-center | grep -v grep 
  2. root 21711 1 1 14:47 pts/3 00:02:10 java -jar mrf-center.jar 

得到進(jìn)程ID為21711,第二步找出該進(jìn)程內(nèi)最耗費(fèi)CPU的線(xiàn)程,可以使用ps -Lfp pid或者ps -mp pid -o THREAD, tid, time或者top -Hp pid,我這里用第三個(gè),輸出如下:

 

JVM性能調(diào)優(yōu)監(jiān)控工具使用詳解

 

TIME列就是各個(gè)Java線(xiàn)程耗費(fèi)的CPU時(shí)間,CPU時(shí)間最長(zhǎng)的是線(xiàn)程ID為21742的線(xiàn)程,用

  1. printf "%x\n" 21742 

得到21742的十六進(jìn)制值為54ee,下面會(huì)用到。

OK,下一步終于輪到j(luò)stack上場(chǎng)了,它用來(lái)輸出進(jìn)程21711的堆棧信息,然后根據(jù)線(xiàn)程ID的十六進(jìn)制值grep,如下:

  1. root@ubuntu:/# jstack 21711 | grep 54ee 
  2. "PollIntervalRetrySchedulerThread" prio=10 tid=0x00007f950043e000 nid=0x54ee in Object.wait [0x00007f94c6eda000] 

可以看到CPU消耗在PollIntervalRetrySchedulerThread這個(gè)類(lèi)的Object.wait,我找了下我的代碼,定位到下面的代碼:

  1. // Idle wait 
  2. getLog.info("Thread [" + getName() + "] is idle waiting..."); 
  3. schedulerThreadState = PollTaskSchedulerThreadState.IdleWaiting; 
  4. long now = System.currentTimeMillis; 
  5. long waitTime = now + getIdleWaitTime; 
  6. long timeUntilContinue = waitTime - now; 
  7. synchronized(sigLock) {try { 
  8. if(!halted.get) { 
  9. sigLock.wait(timeUntilContinue); 
  10. } catch (InterruptedException ignore) { 

它是輪詢(xún)?nèi)蝿?wù)的空閑等待代碼,上面的sigLock.wait(timeUntilContinue)就對(duì)應(yīng)了前面的Object.wait。

C、jmap(Memory Map)和jhat(Java Heap Analysis Tool)

jmap用來(lái)查看堆內(nèi)存使用狀況,一般結(jié)合jhat使用。

jmap語(yǔ)法格式如下:

  1. jmap [option] pid 
  2. jmap [option] executable core 
  3. jmap [option] [server-id@]remote-hostname-or-ip 

如果運(yùn)行在64位JVM上,可能需要指定-J-d64命令選項(xiàng)參數(shù)。

  1. jmap -permstat pid 

打印進(jìn)程的類(lèi)加載器和類(lèi)加載器加載的持久代對(duì)象信息,輸出:類(lèi)加載器名稱(chēng)、對(duì)象是否存活(不可靠)、對(duì)象地址、父類(lèi)加載器、已加載的類(lèi)大小等信息,如下圖:

 

JVM性能調(diào)優(yōu)監(jiān)控工具使用詳解

 

使用jmap -heap pid查看進(jìn)程堆內(nèi)存使用情況,包括使用的GC算法、堆配置參數(shù)和各代中堆內(nèi)存使用情況。比如下面的例子:

  1. root@ubuntu:/# jmap -heap 21711 
  2. Attaching to process ID 21711, please wait... 
  3. Debugger attached successfully. 
  4. Server compiler detected. 
  5. JVM version is 20.10-b01 
  6.  
  7. using thread-local object allocation. 
  8. Parallel GC with 4 thread(s) 
  9.  
  10. Heap Configuration: 
  11. MinHeapFreeRatio = 40 
  12. MaxHeapFreeRatio = 70 
  13. MaxHeapSize = 2067791872 (1972.0MB) 
  14. NewSize = 1310720 (1.25MB) 
  15. MaxNewSize = 17592186044415 MB 
  16. OldSize = 5439488 (5.1875MB) 
  17. NewRatio = 2 
  18. SurvivorRatio = 8 
  19. PermSize = 21757952 (20.75MB) 
  20. MaxPermSize = 85983232 (82.0MB) 
  21.  
  22. Heap Usage: 
  23. PS Young Generation 
  24. Eden Space
  25. capacity = 6422528 (6.125MB) 
  26. used = 5445552 (5.1932830810546875MB) 
  27. free = 976976 (0.9317169189453125MB) 
  28. 84.78829520089286% used 
  29. From Space
  30. capacity = 131072 (0.125MB) 
  31. used = 98304 (0.09375MB) 
  32. free = 32768 (0.03125MB) 
  33. 75.0% used 
  34. To Space
  35. capacity = 131072 (0.125MB) 
  36. used = 0 (0.0MB) 
  37. free = 131072 (0.125MB) 
  38. 0.0% used 
  39. PS Old Generation 
  40. capacity = 35258368 (33.625MB) 
  41. used = 4119544 (3.9287033081054688MB) 
  42. free = 31138824 (29.69629669189453MB) 
  43. 11.683876009235595% used 
  44. PS Perm Generation 
  45. capacity = 52428800 (50.0MB) 
  46. used = 26075168 (24.867218017578125MB) 
  47. free = 26353632 (25.132781982421875MB) 
  48. 49.73443603515625% used 
  49. .... 

使用jmap -histo[:live] pid查看堆內(nèi)存中的對(duì)象數(shù)目、大小統(tǒng)計(jì)直方圖,如果帶上live則只統(tǒng)計(jì)活對(duì)象,如下:

  1. root@ubuntu:/# jmap -histo:live 21711 | more  
  2. num #instances #bytes class name---------------------------------------------- 
  3. 1: 38445 5597736 <constMethodKlass> 
  4. 2: 38445 5237288 <methodKlass> 
  5. 3: 3500 3749504 <constantPoolKlass> 
  6. 4: 60858 3242600 <symbolKlass> 
  7. 5: 3500 2715264 <instanceKlassKlass> 
  8. 6: 2796 2131424 <constantPoolCacheKlass> 
  9. 7: 5543 1317400 [I 
  10. 8: 13714 1010768 [C 
  11. 9: 4752 1003344 [B 
  12. 10: 1225 639656 <methodDataKlass> 
  13. 11: 14194 454208 java.lang.String 
  14. 12: 3809 396136 java.lang.Class 
  15. 13: 4979 311952 [S 
  16. 14: 5598 287064 [[I 
  17. 15: 3028 266464 java.lang.reflect.Method 
  18. 16: 280 163520 <objArrayKlassKlass> 
  19. 17: 4355 139360 java.util.HashMap$Entry 
  20. 18: 1869 138568 [Ljava.util.HashMap$Entry; 
  21. 19: 2443 97720 java.util.LinkedHashMap$Entry 
  22. 20: 2072 82880 java.lang.ref.SoftReference 
  23. 21: 1807 71528 [Ljava.lang.Object; 
  24. 22: 2206 70592 java.lang.ref.WeakReference 
  25. 23: 934 52304 java.util.LinkedHashMap 
  26. 24: 871 48776 java.beans.MethodDescriptor 
  27. 25: 1442 46144 java.util.concurrent.ConcurrentHashMap$HashEntry 
  28. 26: 804 38592 java.util.HashMap 
  29. 27: 948 37920 java.util.concurrent.ConcurrentHashMap$Segment 
  30. 28: 1621 35696 [Ljava.lang.Class; 
  31. 29: 1313 34880 [Ljava.lang.String; 
  32. 30: 1396 33504 java.util.LinkedList$Entry 
  33. 31: 462 33264 java.lang.reflect.Field 
  34. 32: 1024 32768 java.util.Hashtable$Entry 
  35. 33: 948 31440 [Ljava.util.concurrent.ConcurrentHashMap$HashEntry; 

class name是對(duì)象類(lèi)型,說(shuō)明如下:

  1. B byte 
  2. char 
  3. double 
  4. float 
  5. int 
  6. J long 
  7. Z boolean 
  8. [ 數(shù)組,如[I表示int[] 
  9. [L+類(lèi)名 其他對(duì)象 

還有一個(gè)很常用的情況是:用jmap把進(jìn)程內(nèi)存使用情況dump到文件中,再用jhat分析查看。jmap進(jìn)行dump命令格式如下:

  1. jmap -dump:format=b,file=dumpFileName pid 

我一樣地對(duì)上面進(jìn)程ID為21711進(jìn)行Dump:

  1. root@ubuntu:/# jmap -dump:format=b,file=/tmp/dump.dat 21711  
  2. Dumping heap to /tmp/dump.dat ... 
  3. Heap dump file created 

dump出來(lái)的文件可以用MAT、VisualVM等工具查看,這里用jhat查看:

  1. root@ubuntu:/# jhat -port 9998 /tmp/dump.dat 
  2. Reading from /tmp/dump.dat... 
  3. Dump file created Tue Jan 28 17:46:14 CST 2014Snapshot read, resolving... 
  4. Resolving 132207 objects... 
  5. Chasing references, expect 26 dots.......................... 
  6. Eliminating duplicate references.......................... 
  7. Snapshot resolved. 
  8. Started HTTP server on port 9998Server is ready. 

注意如果Dump文件太大,可能需要加上-J-Xmx512m這種參數(shù)指定最大堆內(nèi)存,即jhat -J-Xmx512m -port 9998 /tmp/dump.dat。然后就可以在瀏覽器中輸入主機(jī)地址:9998查看了:

 

JVM性能調(diào)優(yōu)監(jiān)控工具使用詳解

 

上面紅線(xiàn)框出來(lái)的部分大家可以自己去摸索下,最后一項(xiàng)支持OQL(對(duì)象查詢(xún)語(yǔ)言)。

D、jstat(JVM統(tǒng)計(jì)監(jiān)測(cè)工具)

語(yǔ)法格式如下:

  1. jstat [ generalOption | outputOptions vmid [interval[s|ms] [count]] ] 

vmid是Java虛擬機(jī)ID,在Linux/Unix系統(tǒng)上一般就是進(jìn)程ID。interval是采樣時(shí)間間隔。count是采樣數(shù)目。比如下面輸出的是GC信息,采樣時(shí)間間隔為250ms,采樣數(shù)為4:

  1. root@ubuntu:/# jstat -gc 21711 250 4  
  2. S0C S1C S0U S1U EC EU OC OU PC PU YGC YGCT FGC FGCT GCT 
  3. 192.0 192.0 64.0 0.0 6144.0 1854.9 32000.0 4111.6 55296.0 25472.7 702 0.431 3 0.218 0.649 
  4. 192.0 192.0 64.0 0.0 6144.0 1972.2 32000.0 4111.6 55296.0 25472.7 702 0.431 3 0.218 0.649 
  5. 192.0 192.0 64.0 0.0 6144.0 1972.2 32000.0 4111.6 55296.0 25472.7 702 0.431 3 0.218 0.649 
  6. 192.0 192.0 64.0 0.0 6144.0 2109.7 32000.0 4111.6 55296.0 25472.7 702 0.431 3 0.218 0.649 

要明白上面各列的意義,先看JVM堆內(nèi)存布局:

 

JVM性能調(diào)優(yōu)監(jiān)控工具使用詳解

 

可以看出:

  1. 堆內(nèi)存 = 年輕代 + 年老代 + 永久代 
  2. 年輕代 = Eden區(qū) + 兩個(gè)Survivor區(qū)(FromTo

現(xiàn)在來(lái)解釋各列含義:

  1. S0C、S1C、S0U、S1U:Survivor 0/1區(qū)容量(Capacity)和使用量(Used) 
  2. EC、EU:Eden區(qū)容量和使用量 
  3. OC、OU:年老代容量和使用量 
  4. PC、PU:永久代容量和使用量 
  5. YGC、YGT:年輕代GC次數(shù)和GC耗時(shí) 
  6. FGC、FGCT:Full GC次數(shù)和Full GC耗時(shí) 
  7. GCT:GC總耗時(shí) 

E、hprof(Heap/CPU Profiling Tool)

hprof能夠展現(xiàn)CPU使用率,統(tǒng)計(jì)堆內(nèi)存使用情況。

語(yǔ)法格式如下:

  1. java -agentlib:hprof[=options] ToBeProfiledClass 
  2. java -Xrunprof[:options] ToBeProfiledClass 
  3. javac -J-agentlib:hprof[=options] ToBeProfiledClass 

完整的命令選項(xiàng)如下:

  1. Option Name and Value Description Default 
  2. --------------------- ----------- ------- 
  3. heap=dump|sites|all heap profiling all 
  4. cpu=samples|times|old CPU usage off 
  5. monitor=y|n monitor contention n 
  6. format=a|b text(txt) or binary output a 
  7. file=<file> write data to file java.hprof[.txt] 
  8. net=<host>:<port> send data over a socket off 
  9. depth=<size> stack trace depth 4 
  10. interval=<ms> sample interval in ms 10 
  11. cutoff=<value> output cutoff point 0.0001 
  12. lineno=y|n line number in traces? y 
  13. thread=y|n thread in traces? n 
  14. doe=y|n dump on exit? y 
  15. msa=y|n Solaris micro state accounting n 
  16. force=y|n force output to <file> y 
  17. verbose=y|n print messages about dumps y 

來(lái)幾個(gè)官方指南上的實(shí)例。

CPU Usage Sampling Profiling(cpu=samples)的例子:

  1. java -agentlib:hprof=cpu=samples,interval=20,depth=3 Hello 

上面每隔20毫秒采樣CPU消耗信息,堆棧深度為3,生成的profile文件名稱(chēng)是java.hprof.txt,在當(dāng)前目錄。

CPU Usage Times Profiling(cpu=times)的例子,它相對(duì)于CPU Usage Sampling Profile能夠獲得更加細(xì)粒度的CPU消耗信息,能夠細(xì)到每個(gè)方法調(diào)用的開(kāi)始和結(jié)束,它的實(shí)現(xiàn)使用了字節(jié)碼注入技術(shù)(BCI):

  1. javac -J-agentlib:hprof=cpu=times Hello.java 

Heap Allocation Profiling(heap=sites)的例子:

  1. javac -J-agentlib:hprof=heap=sites Hello.java 

Heap Dump(heap=dump)的例子,它比上面的Heap Allocation Profiling能生成更詳細(xì)的Heap Dump信息:

  1. javac -J-agentlib:hprof=heap=dump Hello.java 

 

 

責(zé)任編輯:武曉燕 來(lái)源: 今日頭條
相關(guān)推薦

2017-07-21 08:55:13

TomcatJVM容器

2020-11-09 07:34:49

JVM性能監(jiān)控

2024-12-04 15:49:29

2017-10-17 14:02:30

jvm調(diào)優(yōu)工具

2023-04-24 14:54:09

JVM性能調(diào)優(yōu)

2012-01-10 14:35:08

JavaJVM

2019-02-19 10:25:28

JVM性能工具

2021-12-06 11:03:57

JVM性能調(diào)優(yōu)

2023-11-11 19:07:23

JVMJava

2023-02-10 09:28:23

優(yōu)化工具

2010-09-27 10:20:09

JVMLinux

2010-09-26 11:22:22

JVM垃圾回收JVM

2023-11-28 08:43:48

2021-03-17 11:35:11

JVM代碼Java

2010-09-26 13:39:46

JVM調(diào)優(yōu)

2010-09-26 09:08:17

JVM調(diào)優(yōu)

2012-01-10 15:13:56

JavaJVM

2009-04-01 11:17:52

內(nèi)存分配調(diào)優(yōu)Oracle

2023-03-03 08:14:33

JavaJDK調(diào)優(yōu)

2012-06-20 11:05:47

性能調(diào)優(yōu)攻略
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)