本系列筆記主要基於《深入理解Java虛擬機:JVM高級特性與最佳實踐 第2版》,是這本書的讀書筆記。
jstack命令可以打印Java進程的各個線程堆棧跟蹤信息,可以用來查看Java中各個線程的執行情況,可以用來定位和解決死循環和死鎖導致的CPU飆高的問題。
所有的JDK工具都可以在Oracle官網的 Java Tools Reference 文檔中找到使用說明,這是主要參考,包括命令格式、參數內容、輸出信息等等。
jstack命令格式:
jstack [-l] <pid>
jstack -F [-m] [-l] <pid>
jstack [-m] [-l] <executable> <core>
jstack [-m] [-l] [server_id@]<remote server IP or hostname>
jstack幫助信息:
jstack輸出內容格式
先使用jps命令查詢Java進程,輸出pid。
jps -l
然后使用jstack加上pid,輸出內容到txt文件中。
jstack 42859 > 42859.txt
然后在XShell中可以使用sz命令下載txt文件,也可以在Linux中直接打開。
jstack輸出的內容舉例如下:
"ajp-nio-8009-Acceptor-0" #180 daemon prio=5 os_prio=0 tid=0x00007f279865a000 nid=0xa825 runnable [0x00007f265e0c8000]
java.lang.Thread.State: RUNNABLE
at sun.nio.ch.ServerSocketChannelImpl.accept0(Native Method)
at sun.nio.ch.ServerSocketChannelImpl.accept(ServerSocketChannelImpl.java:422)
at sun.nio.ch.ServerSocketChannelImpl.accept(ServerSocketChannelImpl.java:250)
- locked <0x00000006c0234640> (a java.lang.Object)
at org.apache.tomcat.util.net.NioEndpoint$Acceptor.run(NioEndpoint.java:692)
at java.lang.Thread.run(Thread.java:745)
- "ajp-nio-8009-Acceptor-0":線程名
- "#180":線程編號
- "daemon":后台守護線程
- "prio=5":優先級
- "os_prio=0":操作系統優先級
- "tid=0x00007f279865a000":線程id
- "nid=0xa825":比較關鍵,操作系統映射的線程id(十六進制), 轉成十進制后,對應Linux下
ps -mp pid -o THREAD,tid,time
命令打印出來的線程tid - "0x00007f265e0c8000":線程棧的起始地址
- "RUNNABLE":一種線程狀態
Java線程狀態
Java線程的狀態分為六種,Oracle官網介紹:Thread States for a Thread Dump
- NEW:新建,線程創建后還未啟動,還沒調用start方法。
- RUNNABLE:運行,處於可運行狀態,正在運行或准備運行。
- BLOCKED:阻塞,線程掛起,等待獲取鎖。
- WAITING:無限期等待,線程無限期地等待另一個線程執行特定操作。
- TIMED_WAITING:有限期等待,線程正在等待另一個線程執行最多指定等待時間的操作。
- TERMINATED:線程終止狀態。
Java線程狀態圖示:
首先,當new了一個線程對象時,就是NEW狀態,例如Thread thread = new Thread();
當線程調用了start()方法后,進入RUNNABLE狀態,RUNNABLE中又分為Ready和Running,當獲取到CPU時間片的時候是Running,當等待獲取CPU時間片的時候是Ready。
RUNNABLE狀態時,當線程調用了sleep(time)、wait(time)等方法后,進入TIMED_WAITING狀態,time時間到期后變為RUNNABLE狀態,或者調用notify、notifyAll方法后變為BLOCKED狀態。
RUNNABLE狀態時,當線程調用了sleep()、wait()等方法后,進入WAITING狀態,在調用了notify、notifyAll方法后變為BLOCKED狀態。
RUNNABLE狀態時,當等待獲取鎖時,進入BLOCKED狀態,當獲取到鎖之后,再恢復到RUNNABLE狀態。