JVM問題定位及調優命令 jhat

jhat:

JVM Heap Analysis Tool命令是與jmap搭配使用,用來分析jmap生成的dump,jhat內置了一個微型的HTTP/HTML服務器,生成dump的分析結果後,能夠在瀏覽器中查看。在此要注意,通常不會直接在服務器上進行分析,由於jhat是一個耗時而且耗費硬件資源的過程,通常把服務器生成的dump文件複製到本地或其餘機器上進行分析。【內存分析】java

[work@16-11-118 ~]$ jhat -help
Usage:  jhat [-stack <bool>] [-refs <bool>] [-port <port>] [-baseline <file>] [-debug <int>] [-version] [-h|-help] <file>

	-J<flag>          Pass <flag> directly to the runtime system. For
			  example, -J-mx512m to use a maximum heap size of 512MB
	-stack false:     Turn off tracking object allocation call stack.
	-refs false:      Turn off tracking of references to objects
	-port <port>:     Set the port for the HTTP server.  Defaults to 7000
	-exclude <file>:  Specify a file that lists data members that should
			  be excluded from the reachableFrom query.
	-baseline <file>: Specify a baseline object dump.  Objects in
			  both heap dumps with the same ID and same class will
			  be marked as not being "new".
	-debug <int>:     Set debug level.
			    0:  No debug output
			    1:  Debug hprof file parsing
			    2:  Debug hprof file parsing, no server
	-version          Report version number
	-h|-help          Print this help and exit
	<file>            The file to read

For a dump file that contains multiple heap dumps,
you may specify which dump in the file
by appending "#<number>" to the file name, i.e. "foo.hprof#3".

All boolean options default to "true"

參數web

-J< flag >                 瀏覽器

由於 jhat 命令實際上會啓動一個JVM來執行, 經過 -J 能夠在啓動JVM時傳入一些啓動參數. 例如, -J-Xmx512m 則指定運行 jhat 的Java虛擬機使用的最大堆內存爲 512 MB. 若是須要使用多個JVM啓動參數,則傳入多個 -Jxxxxxx.bash

-stack false|true 服務器

關閉對象分配調用棧跟蹤(tracking object allocation call stack)。 若是分配位置信息在堆轉儲中不可用. 則必須將此標誌設置爲 false. 默認值爲 true.
-refs false|true 網絡

關閉對象引用跟蹤(tracking of references to objects)。 默認值爲 true. 默認狀況下, 返回的指針是指向其餘特定對象的對象,如反向連接或輸入引用(referrers or incoming references), 會統計/計算堆中的全部對象。
-port port-number app

設置 jhat HTTP server 的端口號. 默認值 7000。this

-exclude exclude-file spa

指定對象查詢時須要排除的數據成員列表文件(a file that lists data members that should be excluded from the reachable objects query)。 例如, 若是文件列列出了 java.lang.String.value , 那麼當從某個特定對象 Object o 計算可達的對象列表時, 引用路徑涉及 java.lang.String.value 的都會被排除。
-baseline exclude-file debug

指定一個基準堆轉儲(baseline heap dump)。 在兩個 heap dumps 中有相同 object ID 的對象會被標記爲不是新的(marked as not being new). 其餘對象被標記爲新的(new). 在比較兩個不一樣的堆轉儲時頗有用。
-debug int 

設置 debug 級別. 0 表示不輸出調試信息。 值越大則表示輸出更詳細的 debug 信息。
-version 

啓動後只顯示版本信息就退出。

jhat -J-Xmx512m dump.hprof
jhat -port 7000 mem.dat

jmap -dump:format=b,file=mem.dat pid #將內存使用的詳細狀況輸出到mem.dat 文件
經過jhat -port 7000 mem.dat能夠將mem.dat的內容以web的方式暴露到網絡,訪問http://ip-server:7000查看。

通常查看堆異常狀況主要看這個兩個部分: Show instance counts for all classes (excluding platform),平臺外的全部對象信息。Show heap histogram 以樹狀圖形式展現堆狀況。

 

觀察是否大量應該被回收的對象在一直被引用或者是否有佔用內存特別大的對象沒法被回收。

能夠查看對象的引用關係:例如查看Teacher的引用關係

翻到頁面底部:

以下爲static應用 靜態引用 字段名稱爲 tea

以下爲成員變量引用:DoMain中的teacher成員變量和DoMain2中tt成員變量

上圖每個紅框都爲一個引用堆棧信息:

teacher下直接爲Teacher ,說明teacher爲Teacher類型。

tt下爲ArrayList ,list下爲Object/Teacher類型,說明Object保持Teacher引用 ArrayList保持Object的引用。tt爲ArrayList類型。 針對這個結論咱們來進行驗證:

咱們發現上兩圖 ArrayList內存地址是同樣的。咱們再看咱們定義的實體:

發現tt 的確是 ArrayList類型 咱們認證結論成立。

通過屢次實現,發現局部變量和靜態變量的引用傳遞能夠追溯到,可是局部變量的引用關係追溯不到

可是咱們能夠從具體對象引用追溯信息:針對以下代碼運行

隨便點擊以下連接:

咱們查看該對象的引用:

咱們猜測field爲elementData 時 爲ArrayList對象引用

咱們將其地址信息(java.util.ArrayList@0x7181abce0 或者 0x7181abce0 )複製出來在

Student類引用信息中查找:發現找到其引用地址

往下看,發現是在DoMain中有其局部變量引用.能夠定位代碼。

相關文章
相關標籤/搜索