當(dāng)前位置:首頁 > 公眾號精選 > 架構(gòu)師社區(qū)
[導(dǎo)讀]來自:冰河技術(shù)?? 寫在前面 在【高并發(fā)專題】中的《高并發(fā)之——線程與多線程》一文中,我們簡單介紹了線程的生命周期和線程的幾個重要狀態(tài),并以代碼的形式實現(xiàn)了線程是如何進入各個狀態(tài)的。今天,我們就結(jié)合操作系統(tǒng)線程和編程語言線程再次深入探討線程的生

線程的生命周期其實沒有我們想象的那么簡單!!

來自:冰河技術(shù)  

寫在前面

在【高并發(fā)專題】中的《高并發(fā)之——線程與多線程》一文中,我們簡單介紹了線程的生命周期和線程的幾個重要狀態(tài),并以代碼的形式實現(xiàn)了線程是如何進入各個狀態(tài)的。今天,我們就結(jié)合操作系統(tǒng)線程和編程語言線程再次深入探討線程的生命周期問題,線程的生命周期其實沒有我們想象的那么簡單??!

理解線程的生命周期本質(zhì)上理解了生命周期中各個節(jié)點的狀態(tài)轉(zhuǎn)換機制就可以了。

接下來,我們分別就通用線程生命周期和Java語言的線程生命周期分別進行詳細說明。

通用的線程生命周期

通用的線程生命周期總體上可以分為五個狀態(tài):初始狀態(tài)、可運行狀態(tài)、運行狀態(tài)、休眠狀態(tài)和終止?fàn)顟B(tài)。

我們可以簡單的使用下圖來表示這五種狀態(tài)。

線程的生命周期其實沒有我們想象的那么簡單??!

初始狀態(tài)

線程已經(jīng)被創(chuàng)建,但是不允許分配CPU執(zhí)行。需要注意的是:這個狀態(tài)屬于編程語言特有,這里指的線程已經(jīng)被創(chuàng)建,僅僅指在編程語言中被創(chuàng)建,在操作系統(tǒng)中,并沒有創(chuàng)建真正的線程。

可運行狀態(tài)

線程可以分配CPU執(zhí)行。此時,操作系統(tǒng)中的線程被成功創(chuàng)建,可以分配CPU執(zhí)行。

運行狀態(tài)

當(dāng)操作系統(tǒng)中存在空閑的CPU,操作系統(tǒng)會將這個空閑的CPU分配給一個處于可運行狀態(tài)的線程,被分配到CPU的線程的狀態(tài)就轉(zhuǎn)換成了運行狀態(tài)

休眠狀態(tài)

運行狀態(tài)的線程調(diào)用一個阻塞的API(例如,以阻塞的方式讀文件)或者等待某個事件(例如,等待某個條件變量等),線程的狀態(tài)就會轉(zhuǎn)換到休眠狀態(tài)。此時線程會釋放CPU資源,休眠狀態(tài)的線程沒有機會獲得CPU的使用權(quán)。一旦等待的條件出現(xiàn),線程就會從休眠狀態(tài)轉(zhuǎn)換到可運行狀態(tài)。

終止?fàn)顟B(tài)

線程執(zhí)行完畢或者出現(xiàn)異常就會進入終止?fàn)顟B(tài),終止?fàn)顟B(tài)的線程不會切換到其他任何狀態(tài),這也意味著線程的生命周期結(jié)束了。

以上就是通用的線程生命周期,下面,我們再看對比看下Java語言中的線程生命周期。

Java中的線程生命周期

Java中的線程生命周期總共可以分為六種狀態(tài):初始化狀態(tài)(NEW)、可運行/運行狀態(tài)(RUNNABLE)、阻塞狀態(tài)(BLOCKED)、無時限等待狀態(tài)(WAITING)、有時限等待狀態(tài)(TIMED_WAITING)、終止?fàn)顟B(tài)(TERMINATED)。

需要大家重點理解的是:雖然Java語言中線程的狀態(tài)比較多,但是,其實在操作系統(tǒng)層面,Java線程中的阻塞狀態(tài)(BLOCKED)、無時限等待狀態(tài)(WAITING)、有時限等待狀態(tài)(TIMED_WAITING)都是一種狀態(tài),即通用線程生命周期中的休眠狀態(tài)。也就是說,只要Java中的線程處于這三種狀態(tài)時,那么,這個線程就沒有CPU的使用權(quán)。

理解了這些之后,我們就可以使用下面的圖來簡單的表示Java中線程的生命周期。

線程的生命周期其實沒有我們想象的那么簡單??!

我們也可以這樣理解阻塞狀態(tài)(BLOCKED)、無時限等待狀態(tài)(WAITING)、有時限等待狀態(tài)(TIMED_WAITING),它們是導(dǎo)致線程休眠的三種原因!

接下來,我們就看看Java線程中的狀態(tài)是如何轉(zhuǎn)化的。

RUNNABLE與BLOCKED的狀態(tài)轉(zhuǎn)換

只有一種場景會觸發(fā)這種轉(zhuǎn)換,就是線程等待synchronized隱式鎖。synchronized修飾的方法、代碼塊同一時刻只允許一個線程執(zhí)行,其他的線程則需要等待。此時,等待的線程就會從RUNNABLE狀態(tài)轉(zhuǎn)換到BLOCKED狀態(tài)。當(dāng)?shù)却木€程獲得synchronized隱式鎖時,就又會從BLOCKED狀態(tài)轉(zhuǎn)換到RUNNABLE狀態(tài)。

這里,需要大家注意:線程調(diào)用阻塞API時,在操作系統(tǒng)層面,線程會轉(zhuǎn)換到休眠狀態(tài)。但是在JVM中,Java線程的狀態(tài)不會發(fā)生變化,也就是說,Java線程的狀態(tài)仍然是RUNNABLE狀態(tài)。JVM并不關(guān)心操作系統(tǒng)調(diào)度相關(guān)的狀態(tài),在JVM角度來看,等待CPU使用權(quán)(操作系統(tǒng)中的線程處于可執(zhí)行狀態(tài))和等待IO操作(操作系統(tǒng)中的線程處于休眠狀態(tài))沒有區(qū)別,都是在等待某個資源,所以,將其都歸入了RUNNABLE狀態(tài)。

我們平時所說的Java在調(diào)用阻塞API時,線程會阻塞,指的是操作系統(tǒng)線程的狀態(tài),并不是Java線程的狀態(tài)。

RUNNABLE與WAITING狀態(tài)轉(zhuǎn)換

線程從RUNNABLE狀態(tài)轉(zhuǎn)換成WAITING狀態(tài)總體上有三種場景。

場景一

獲得synchronized隱式鎖的線程,調(diào)用無參的Object.wait()方法。此時的線程會從RUNNABLE狀態(tài)轉(zhuǎn)換成WAITING狀態(tài)。

場景二

調(diào)用無參數(shù)的Thread.join()方法。其中join()方法是一種線程的同步方法。例如,在threadA線程中調(diào)用threadB線程的join()方法,則threadA線程會等待threadB線程執(zhí)行完。而threadA線程在等待threadB線程執(zhí)行的過程中,其狀態(tài)會從RUNNABLE轉(zhuǎn)換到WAITING。當(dāng)threadB執(zhí)行完畢,threadA線程的狀態(tài)則會從WAITING狀態(tài)轉(zhuǎn)換成RUNNABLE狀態(tài)。

場景三

調(diào)用LockSupport.park()方法,當(dāng)前線程會阻塞,線程的狀態(tài)會從RUNNABLE轉(zhuǎn)換成WAITING。調(diào)用LockSupport.unpark(Thread thread)可喚醒目標(biāo)線程,目標(biāo)線程的狀態(tài)又會從WAITING狀態(tài)轉(zhuǎn)換到RUNNABLE。

RUNNABLE與TIMED_WAITING狀態(tài)轉(zhuǎn)換

總體上可以分為五種場景。

場景一

調(diào)用帶超時參數(shù)的Thread.sleep(long millis)方法;

場景二

獲得synchronized隱式鎖的線程,調(diào)用帶超時參數(shù)的Object.wait(long timeout)參數(shù);

場景三

調(diào)用帶超時參數(shù)的Thread.join(long millis)方法;

場景四

調(diào)用帶超時參數(shù)的LockSupport.parkNanos(Object blocker, long deadline)方法;

場景五

調(diào)用帶超時參數(shù)的LockSuppor.parkUntil(long deadline)方法。

從NEW到RUNNABLE狀態(tài)

Java剛創(chuàng)建出來的Thread對象就是NEW狀態(tài),創(chuàng)建Thread對象主要有兩種方法,一種是繼承Thread對象,重寫run()方法;另一種是實現(xiàn)Runnable接口,重寫run()方法。

注意:這里說的是創(chuàng)建Thread對象的方法,而不是創(chuàng)建線程的方法,創(chuàng)建線程的方法包含創(chuàng)建Thread對象的方法。

繼承Thread對象

public class ChildThread extends Thread{
    @Override
    public void run(){
        //線程中需要執(zhí)行的邏輯
    }
}
//創(chuàng)建線程對象
ChildThread childThread = new ChildThread();

實現(xiàn)Runnable接口

public class ChildRunnable implements Runnable{
    @Override
    public void run(){
        //線程中需要執(zhí)行的邏輯
    }
}
//創(chuàng)建線程對象
Thread childThread = new Thread(new ChildRunnable());

處于NEW狀態(tài)的線程不會被操作系統(tǒng)調(diào)度,因此也就不會執(zhí)行。Java中的線程要執(zhí)行,就需要轉(zhuǎn)換到RUNNABLE狀態(tài)。從NEW狀態(tài)轉(zhuǎn)換到RUNNABLE狀態(tài),只需要調(diào)用線程對象的start()方法即可。

//創(chuàng)建線程對象
Thread childThread = new Thread(new ChildRunnable());
//調(diào)用start()方法使線程從NEW狀態(tài)轉(zhuǎn)換到RUNNABLE狀態(tài)
childThread.start();

RUNNABLE到TERMINATED狀態(tài)

線程執(zhí)行完run()方法后,或者執(zhí)行run()方法的時候拋出異常,都會終止,此時為TERMINATED狀態(tài)。如果我們需要中斷run()方法,可以調(diào)用interrupt()方法。

寫在最后

最后,附上并發(fā)編程需要掌握的核心技能知識圖,祝大家在學(xué)習(xí)并發(fā)編程時,少走彎路。

線程的生命周期其實沒有我們想象的那么簡單??!

特別推薦一個分享架構(gòu)+算法的優(yōu)質(zhì)內(nèi)容,還沒關(guān)注的小伙伴,可以長按關(guān)注一下:

線程的生命周期其實沒有我們想象的那么簡單!!

長按訂閱更多精彩▼

線程的生命周期其實沒有我們想象的那么簡單!!

如有收獲,點個在看,誠摯感謝

免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務(wù)。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉