標籤:java 運行 term 運行時 dump bin sudo 調優 trace
Dump 就是對程式運行時記憶體上的資訊進行轉儲, 讓我們可以查看程式當時的運行情況. Dump 對於調優和排錯是非常有用的工具.
Heap Dump
Java 運行時對象分配在堆記憶體上, Heap dump 就是對堆記憶體進行轉儲.
產生
Heap dump 的產生有兩種方式:
1) 運行 Java 程式時添加 -XX:+HeapDumpOnOutOfMemoryError
選項, 這樣當程式發生 Out of Memory 錯誤時就會自動產生一份 Heap dump.
2) 使用 jmap
工具產生. 首先我們用 jps
找到程式的 pid (嚴謹點說其實是 lvmid), 然後運行:
jmap -dump:live,format=b,file=heap.bin <pid>
分析
可以使用 Java 內建的 jhat
工具來分析 Heap dump:
jhat <heap dump file>
等待一會, 就會提示
Started HTTP server on port 7000Server is ready.
這時候瀏覽器中訪問 127.0.0.1:7000
就可以了.
但是, jhat
在分析較大的 Heap dump 時效率比較差, 所以推薦使用 eclipse 提供的 Memory Analyzer (MAT) 來分析.
Thread Dump
Thread dump 轉儲的是線程相關的記憶體資料 (例如該線程的調用棧). Thread dump 有時候也被成為 javacore, 不過好像 javacore 是 IBM 虛擬機器才有的.
產生
可以使用內建的 jstack
產生 Thread dump:
jstack <pid> >> thread.dump
分析
Thread dump 就是個文字檔格式, 直接開啟查看就可以了.
Intellij IDEA 提供 Stacktrace 的分析, 我們可以用它來分析 Thread dump, 這樣可以方便的知道某個線程運行到哪裡.
開啟 Intellij IDEAD -> Analyze -> Anaylyze Stacktrace...
, 把 Thread dump 的內容複寫粘貼進去, 確認即可.
Core Dump
上面提到的 Heap dump 和 Thread dump 都是和 Java 直接相關的, Core dump 則是作業系統提供的, 所有程式在意外退出時, 作業系統都可以產生 Core dump.
Core dump 包含了程式運行時的所有記憶體資訊, 所以我們可以使用 Core dump 同時分析堆記憶體和運行時棧.
產生
預設作業系統是不產生 Core dump 的, 我們需要先開啟:
# 如果你用的是 bashulimit -c unlimited# 如果你像我一樣用的是 zshlimit coredumpsize unlimited
ulimit/limit 是設定 dump 的大小的, 預設為 0 也就是不 dump. 我們可以使用下面的命令來查看當前設定的大小:
# 如果你用的是 bashulimit -c# 如果你像我一樣用的是 zshlimit coredumpsize
確認開啟後, 我們可以使用 kill -ABRT <pid>
來產生 Core dump. 不過需要注意的是, 使用這種方法只有在當前 Terminal 下啟動並執行 Java 程式才能產生 Core dump. 也就是說, 你必須在開啟了 Core dump 的 Terminal 下運行 Java 程式, 這樣 kill -ABRT <pid>
才會產生 Core dump. 如果你 Java 程式運行在一個沒有開啟 Core dump 的 Terminal 下, 那麼即使你的 kill -ABRT <pid>
運行在開啟了 Core dump 的 Terminal 下, 這時候 Core dump 也是不會產生的.
我們也可以使用 gcore
來產生產生 Core dump. 使用這個方法就無所謂你有沒有使用 ulimit/limit 開啟 Core dump 了.
sudo gcore <pid>
Mac 下 Core dump 產生在 /cores/
檔案夾下.
分析
我們可以使用 gdb
來分析 Core dump 檔案.
Java 內建的 jstack
和 jmap
也可以用來分析 Core dump:
jstack <executable> <core dump file>jmap <executable> <core dump file>
這裡的 <executable>
指的是你運行 Java 程式時使用的 java
, 一般可以用 $JAVA_HOME/bin/java
代替. 如果你指定的 java
和你運行用的 java
不是同一個版本, 就會拋出 sun.jvm.hotspot.debugger.UnmappedAddressException
.
另外你使用的 jstack
和 jamp
也需要是相應的版本, 否則會提示 Can‘t attach to the core file
.
轉自:http://yoncise.com/2017/05/23/Java-Heap-Dump-Thread-Dump-and-Core-Dump/
轉:Java - Heap Dump, Thread Dump and Core Dump