詳解Java的線程狀態(tài)
Java的每個線程都具有自己的狀態(tài),Thread類中成員變量threadStatus存儲了線程的狀態(tài):
private volatile int threadStatus = 0;
在Thread類中也定義了狀態(tài)的枚舉,共六種,如下:
public enum State { NEW, // 新建狀態(tài) RUNNABLE, // 執(zhí)行狀態(tài) BLOCKED, // 阻塞狀態(tài) WAITING, // 無限期等待狀態(tài) TIMED_WAITING, // 有限期等待狀態(tài) TERMINATED; // 退出狀態(tài) }
threadStatus初始值為0,對應(yīng)的就是NEW狀態(tài)。
- NEW:新建狀態(tài),new Thread()時處于這個狀態(tài),此時線程還未開始執(zhí)行
- RUNNABLE:執(zhí)行狀態(tài),當調(diào)用了start方法后,線程處于此狀態(tài),當然此刻CPU不一定正在執(zhí)行它
- BLOCKED:阻塞狀態(tài),線程等待鎖時處于此狀態(tài)
- WAITING:無限期等待狀態(tài),需要被喚醒的等待屬于此狀態(tài),如Object.wait
- TIMED_WAITING:有限期等待狀態(tài),調(diào)用一些有超時時間的等待方法會進入此狀態(tài),如Thread.sleep、Object.wait、Thread.join、Lock.tryLock、Condition.await
- TERMINATED:退出狀態(tài),可能是正常運行完畢,也可能是拋出了異常導致線程終止
我們可以通過getState獲取線程的狀態(tài):
State state = thread.getState();
接下來,我們通過示例來感受線程狀態(tài)的變化。
示例一:
Thread t = new Thread(() -> { System.out.println("sleep"); // 休眠2秒 ThreadUtil.sleep(2000); System.out.println("run"); }); // 啟動前先打印一下線程狀態(tài) System.out.println(t.getState()); // 啟動線程 t.start(); // 啟動后立即打印一次線程狀態(tài) System.out.println(t.getState()); // 每隔500毫秒打印一次線程狀態(tài) while (true) { ThreadUtil.sleep(500); System.out.println(t.getState()); }
輸出:
NEW
RUNNABLE
sleep
TIMED_WAITING
TIMED_WAITING
TIMED_WAITING
run
TERMINATED
TERMINATED
TERMINATED
TERMINATED
TERMINATED
TERMINATED
TERMINATED
可以看出,線程創(chuàng)建后,初始狀態(tài)為NEW;
調(diào)用start方法后狀態(tài)變更為了RUNNABLE;
當線程內(nèi)執(zhí)行了sleep,休眠2秒鐘,狀態(tài)變更為了TIMED_WAITING;
當線程執(zhí)行完成后,狀態(tài)變更為了TERMINATED。
示例二:
本示例演示在多線程爭搶鎖的情況下,線程狀態(tài)的變化。
private static Object lock = new Object(); public static void main(String[] args) { Runnable runnable = () -> { // 加鎖 synchronized (lock) { System.out.println(Thread.currentThread().getName() + " lock"); ThreadUtil.sleep(2000); } System.out.println(Thread.currentThread().getName() + " exit"); }; Thread t1 = new Thread(runnable, "t1"); Thread t2 = new Thread(runnable, "t2"); t1.start(); t2.start(); while (true) { ThreadUtil.sleep(500); System.out.println(t1.getName() + ":" + t1.getState()); System.out.println(t2.getName() + ":" + t2.getState()); } }
輸出:
t1 lock t1獲得了鎖
t1:TIMED_WAITING t1 sleep,進入有限等待狀態(tài)
t2:BLOCKED t2未獲得鎖,因此進入阻塞狀態(tài)
t1:TIMED_WAITING
t2:BLOCKED
t1:TIMED_WAITING
t2:BLOCKED
t1 exit t1釋放鎖,執(zhí)行完成,退出
t2 lock t2獲得鎖
t1:TERMINATED t1已經(jīng)是退出狀態(tài)
t2:TIMED_WAITING t2 sleep,進入有限等待狀態(tài)
t1:TERMINATED
t2:TIMED_WAITING
t1:TERMINATED
t2:TIMED_WAITING
t1:TERMINATED
t2:TIMED_WAITING
t2 exit t2釋放鎖,執(zhí)行完成,退出
t1:TERMINATED t1為退出狀態(tài)
t2:TERMINATED t2為退出狀態(tài)
示例三:
本示例演示了線程如何進入和退出WAITING狀態(tài)。
private static Object obj = new Object(); public static void main(String[] args) { Runnable runnable = () -> { synchronized (obj) { try { // 等待 obj.wait(); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("exit"); } }; Thread t1 = new Thread(runnable, "t1"); System.out.println(t1.getName() + ":" + t1.getState()); t1.start(); System.out.println(t1.getName() + ":" + t1.getState()); for (int i = 0; i < 20; i++) { ThreadUtil.sleep(500); if(i == 3) { synchronized (obj) { // 喚醒 obj.notify(); } } System.out.println(t1.getName() + ":" + t1.getState()); } }
輸出:
t1:NEW
t1:RUNNABLE
t1:WAITING 調(diào)用了obj.wait()進入無限等待狀態(tài)
t1:WAITING
t1:WAITING
t1:BLOCKED 主線程獲得鎖,進行notify,t1進入了阻塞狀態(tài)
exit t1線程退出同步塊,執(zhí)行完成
t1:TERMINATED t1線程退出
t1:TERMINATED
Java Thread的threadStatus字段值的更新代碼位于hotspot c++源碼中,JDK源碼中看不到。
Java的線程狀態(tài)并非是操作系統(tǒng)的實際線程狀態(tài),但與操作系統(tǒng)的線程狀態(tài)是有對應(yīng)關(guān)系的,后續(xù)有需要深入分析操作系統(tǒng)原理、hotspot源碼再展開。
到此這篇關(guān)于詳解Java的線程狀態(tài)的文章就介紹到這了,更多相關(guān)Java線程狀態(tài)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
mybatis返回map類型數(shù)據(jù)空值字段不顯示的解決方案
這篇文章主要介紹了mybatis返回map類型數(shù)據(jù)空值字段不顯示的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-03-03Java volatile關(guān)鍵字原理剖析與實例講解
volatile是Java提供的一種輕量級的同步機制,Java?語言包含兩種內(nèi)在的同步機制:同步塊(或方法)和?volatile?變量,本文將詳細為大家總結(jié)Java volatile關(guān)鍵字,通過詳細的代碼示例給大家介紹的非常詳細,需要的朋友可以參考下2023-07-07JAVA中HTTP基本認證(Basic Authentication)實現(xiàn)
HTTP 基本認證是一種簡單的認證方法,本文主要介紹了JAVA中HTTP基本認證(Basic Authentication),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2024-07-07swagger注解@ApiModelProperty失效情況的解決
這篇文章主要介紹了swagger注解@ApiModelProperty失效情況的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-06-06Java并發(fā)J.U.C并發(fā)容器類list set queue
這篇文章主要為大家介紹了Java并發(fā),J.U.C并發(fā)容器類list set queue,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-06-06Springboot中com.mysql.cj.jdbc.Driver在yml文件中爆紅的原因解讀
這篇文章主要介紹了Springboot中com.mysql.cj.jdbc.Driver在yml文件中爆紅的原因解讀,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-05-05