日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

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

線(xiàn)程的生命周期其實(shí)沒(méi)有我們想象的那么簡(jiǎn)單!!

來(lái)自:冰河技術(shù)  

寫(xiě)在前面

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

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

接下來(lái),我們分別就通用線(xiàn)程生命周期和Java語(yǔ)言的線(xiàn)程生命周期分別進(jìn)行詳細(xì)說(shuō)明。

通用的線(xiàn)程生命周期

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

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

線(xiàn)程的生命周期其實(shí)沒(méi)有我們想象的那么簡(jiǎn)單?。? ></p>
   <h4 style=初始狀態(tài)

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

可運(yùn)行狀態(tài)

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

運(yùn)行狀態(tài)

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

休眠狀態(tài)

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

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

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

以上就是通用的線(xiàn)程生命周期,下面,我們?cè)倏磳?duì)比看下Java語(yǔ)言中的線(xiàn)程生命周期。

Java中的線(xiàn)程生命周期

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

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

理解了這些之后,我們就可以使用下面的圖來(lái)簡(jiǎn)單的表示Java中線(xiàn)程的生命周期。

線(xiàn)程的生命周期其實(shí)沒(méi)有我們想象的那么簡(jiǎn)單!!

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

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

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

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

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

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

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

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

場(chǎng)景一

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

場(chǎng)景二

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

場(chǎng)景三

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

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

總體上可以分為五種場(chǎng)景。

場(chǎng)景一

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

場(chǎng)景二

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

場(chǎng)景三

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

場(chǎng)景四

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

場(chǎng)景五

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

從NEW到RUNNABLE狀態(tài)

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

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

繼承Thread對(duì)象

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

實(shí)現(xiàn)Runnable接口

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

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

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

RUNNABLE到TERMINATED狀態(tài)

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

寫(xiě)在最后

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

線(xiàn)程的生命周期其實(shí)沒(méi)有我們想象的那么簡(jiǎn)單!!

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

線(xiàn)程的生命周期其實(shí)沒(méi)有我們想象的那么簡(jiǎn)單!!

長(zhǎng)按訂閱更多精彩▼

線(xiàn)程的生命周期其實(shí)沒(méi)有我們想象的那么簡(jiǎn)單?。? ></p><p style=如有收獲,點(diǎn)個(gè)在看,誠(chéng)摯感謝

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

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