日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

Java線程池中多余的線程是如何回收的

瀏覽:33日期:2022-08-11 17:45:29

最近閱讀了JDK線程池ThreadPoolExecutor的源碼,對線程池執(zhí)行任務(wù)的流程有了大體了解,實際上這個流程也十分通俗易懂,就不再贅述了,別人寫的比我好多了。

不過,我倒是對線程池是如何回收工作線程比較感興趣,所以簡單分析了一下,加深對線程池的理解吧。

那么,就以JDK1.8為例分析吧。

1.runWorker(Worker w)

工作線程啟動后,就進入runWorker(Worker w)方法。

里面是一個while循環(huán),循環(huán)判斷任務(wù)是否為空,若不為空,執(zhí)行任務(wù);若取不到任務(wù),或發(fā)生異常,退出循環(huán),執(zhí)行processWorkerExit(w, completedAbruptly); 在這個方法里把工作線程移除掉。

取任務(wù)的來源有兩個,一個是firstTask,這個是工作線程第一次跑的時候執(zhí)行的任務(wù),最多只能執(zhí)行一次,后面得從getTask()方法里取任務(wù)??磥恚琯etTask()是關(guān)鍵,在不考慮異常的場景下,返回null,就表示退出循環(huán),結(jié)束線程。下一步,就得看看,什么情況下getTask()會返回null。

(篇幅有限,分段截取,省略中間執(zhí)行任務(wù)的步驟)

Java線程池中多余的線程是如何回收的

Java線程池中多余的線程是如何回收的

2. getTask() 返回null

一共有兩種情況會返回null,見紅框處 。

第一種情況,線程池的狀態(tài)已經(jīng)是STOP,TIDYING, TERMINATED,或者是SHUTDOWN且工作隊列為空;

第二種情況,工作線程數(shù)已經(jīng)大于最大線程數(shù)或當前工作線程已超時,且,還有其他工作線程或任務(wù)隊列為空。這點比較難理解,總之先記住,后面會用。

下面以條件1和條件2分別指代這兩種情況的判斷條件。

Java線程池中多余的線程是如何回收的

3. 分場景分析線程池回收工作線程3.1 未調(diào)用shutdown() ,RUNNING狀態(tài)下全部任務(wù)執(zhí)行完成的場景

這種場景,會將工作線程的數(shù)量減少到核心線程數(shù)大小(如果本來就沒有超過,則不需要回收)。

比如一個線程池,核心線程數(shù)為4,最大線程數(shù)為8。一開始是4個工作線程,當任務(wù)把任務(wù)隊列塞滿,就得將工作線程增加到8. 當后面任務(wù)執(zhí)行到差不多了,線程取不到任務(wù)了,就會回收到4個工作線程的狀態(tài)(取決于allowCoreThreadTimeOut的值,這里討論默認值false的情況,即核心線程不會超時。如果為true,工作線程可以全部銷毀)。

可以先排除上面提到的條件1,線程池的狀態(tài)已經(jīng)是STOP,TIDYING, TERMINATED,或者是SHUTDOWN且工作隊列為空。因為線程池一直是RUNNING,這條判斷永遠是false。在這個場景中,可以當條件1不存在。

下面分析取不出任務(wù)時線程是怎么運行的。

step1. 從任務(wù)隊列取任務(wù)有兩種方式,超時等待還是可以一直阻塞下去。決定因素是timed變量。該變量在前面賦值,如果當前線程數(shù)大于核心線程數(shù),變量timed為true, 否則為false(上面說了,這里只討論allowCoreThreadTimeOut為false的情況)。很明顯,現(xiàn)在討論的是timed為true的情況。keepAliveTime一般不設(shè)置,默認值為0,所以基本上可以認為是不阻塞,馬上返回取任務(wù)的結(jié)果。

在線程超時等待喚醒之后,發(fā)現(xiàn)取不出任務(wù),timeOut變?yōu)閠rue,進入下一次循環(huán)。

step2. 來到條件1的判斷,線程池一直RUNNING, 不進入代碼塊。

step3. 來到條件2的判斷,這時任務(wù)隊列為空,條件成立,CAS減少線程數(shù),若成功,返回null,否則,重復(fù)step1。

這里要注意,有可能多條線程同時通過條件2的判斷,那會不會減少后線程的數(shù)量反而比預(yù)想的核心線程數(shù)少呢?

比如當前線程數(shù)已經(jīng)只有5條了,此時有兩條線程同時喚醒,通過條件2的判斷,同時減少數(shù)量,那剩下的線程數(shù)反而只有3條,和預(yù)期不一致。

實際上是不會的。為了防止這種情況,compareAndDecrementWorkerCount(c) 用的是CAS方法,如果CAS失敗就continue,進入下一輪循環(huán),重新判斷。

像上述例子,其中一條線程會CAS失敗,然后重新進入循環(huán),發(fā)現(xiàn)工作線程數(shù)已經(jīng)只有4了,timed為false, 這條線程就不會被銷毀,可以一直阻塞了(workQueue.take())。

這一點我思考了很久才得出答案,一直在想沒有加鎖的情況下是怎么保證一定能不多不少回收到核心線程數(shù)的呢。原來是CAS的奧妙。

從這里也可以看出,雖然有核心線程數(shù),但線程并沒有區(qū)分是核心還是非核心,并不是先創(chuàng)建的就是核心,超過核心線程數(shù)后創(chuàng)建的就是非核心,最終保留哪些線程,完全隨機。

3.2 調(diào)用shutdown() ,全部任務(wù)執(zhí)行完成的場景

這種場景,無論是核心線程還是非核心線程,所有工作線程都會被銷毀。

在調(diào)用shutdown()之后,會向所有的空閑工作線程發(fā)送中斷信號。

Java線程池中多余的線程是如何回收的

最終傳入false,調(diào)用下面這個方法。

Java線程池中多余的線程是如何回收的

可以看出,在發(fā)出中斷信號前,會判斷是否已經(jīng)中斷,以及要獲得工作線程的獨占鎖。

發(fā)出中斷信號的時候,工作線程要么在getTask()里準備獲取任務(wù),要么在執(zhí)行任務(wù),那就得等它執(zhí)行完當前任務(wù)才會發(fā)出,因為工作線程在執(zhí)行任務(wù)的時候,也會工作線程加鎖。工作線程執(zhí)行完任務(wù),又跑到getTask()里面去了。

所以我們只要看getTask()里面怎么應(yīng)對中斷異常的就可以了。

Java線程池中多余的線程是如何回收的

工作線程在getTask()里,有兩種可能。

3.2.1 任務(wù)已全部完成,線程在阻塞等待。

很簡單,中斷信號將其喚醒,從而進入下一輪循環(huán)。到達條件1處,符合條件,減少工作線程數(shù)量,并返回null,由外層結(jié)束這條線程。

這里的decrementWorkerCount()是自旋式的,一定會減1。

Java線程池中多余的線程是如何回收的

3.2.2 任務(wù)還沒有完全執(zhí)行完

調(diào)用shutdown()之后,未執(zhí)行完的任務(wù)要執(zhí)行完畢,池子才能結(jié)束。所以此時有可能線程還在工作。

這里又要分兩個階段討論

階段1 任務(wù)較多,工作線程都能獲得任務(wù)

這里還不涉及到線程退出,可以跳過不看,只是分析一下收到中斷信號后線程的表現(xiàn)。

假設(shè)有線程A,正通過getTask()里獲取任務(wù)。此時A被中斷,在獲取任務(wù)時,無論是poll()還是take(),都會拋出中斷異常。異常被捕獲,重新進入下一輪循環(huán),只要隊列不為空,就可以繼續(xù)取任務(wù)。

線程A被中斷,再次取任務(wù),調(diào)用workQueue.poll() or workQueue.take(),不會拋出異常嗎?還可以正常取出任務(wù)嗎?

這就要看workQueue的實現(xiàn)了。workQueue是BlockingQueue類型,以常見的LinkedBlockingQueue和ArrayBlockingQueue為例,加鎖時都是調(diào)用lockInterruptibly(),是響應(yīng)中斷的。該方法又調(diào)用了AQS的acquireInterruptibly(int arg)。

acquireInterruptibly(int arg),無論是在入口處判斷中斷異常,還是在parkAndCheckInterrupt()方法阻塞,被中斷喚醒并判斷中斷異常時,均使用了Thread.interrupted()。這個方法會返回線程的中斷狀態(tài),并把中斷狀態(tài)重置!也就是說,線程不再是中斷狀態(tài)了,這樣在再次取任務(wù)時,就不會報錯了。

因此,這對于正在準備取任務(wù)的線程,只是相當于浪費了一次循環(huán),這可能是線程中斷帶來的副作用吧,當然,對整體的運行不影響。

分析到這里,我不禁感嘆,這里BlockingQueue剛好是會重置中斷狀態(tài),這到底是怎么想出來的絕妙設(shè)計???Doug Lea大神Orz.

Java線程池中多余的線程是如何回收的

Java線程池中多余的線程是如何回收的

階段2 任務(wù)剛好要執(zhí)行完了

這時任務(wù)已經(jīng)快取完了,比如有4條工作線程,只剩下2個任務(wù),那就可能出現(xiàn)2條線程獲得任務(wù),2條線程阻塞。

因為在獲取任務(wù)前的判斷,沒有加鎖,那么會不會出現(xiàn),所有線程都通過了前面的校驗,來到workQueue獲取任務(wù)的地方,剛好任務(wù)隊列已經(jīng)空了,線程全部阻塞了呢?因為shutdown() 已經(jīng)執(zhí)行完畢,無法再向線程發(fā)出中斷信號,從而線程一直在阻塞,無法被回收。

這種是不會發(fā)生的。

假設(shè)有A,B,C,D四條工作線程,同時通過了條件1和條件2的判斷,來到取任務(wù)的地方。那么,工作隊列至少還有一個任務(wù),至少會有一條線程能取到任務(wù)。

假設(shè)A,B獲得了任務(wù),C,D阻塞。

A, B接下來的步驟是:

step1.任務(wù)執(zhí)行完成后,再次getTask(),此時符合條件1,返回null,線程準備被回收。

step2.processWorkerExit(Worker w, boolean completedAbruptly) 將線程回收。

回收就只是把線程干掉這么簡單嗎?來看看processWorkerExit(Worker w, boolean completedAbruptly) 的方法。

Java線程池中多余的線程是如何回收的

可以看到,在里面除了workers.remove(w) 移除線,還調(diào)用了tryTerminate()。

Java線程池中多余的線程是如何回收的

第一個判斷條件沒有一個子條件符合,跳過。第二個條件,工作線程還存在,那么隨機中斷一條空閑線程。

那么問題就來了,中斷一條空閑線程,也沒說是一定中斷正在阻塞的線程啊。如果A, B同時退出,有沒有可能出現(xiàn)A中斷B, B中斷A,AB互相中斷,從而沒有線程去中斷喚醒阻塞的線程呢?

答案仍然是,想多了……

假設(shè)A能走到這里,說明A已經(jīng)從工作線程的集合workers里面移除了(processWorkerExit(Worker w, boolean completedAbruptly) 在tryTerminate()之前,已經(jīng)將其移除)。那么A中斷B,B來到這里中斷,就不會在workers里面找到A了。

Java線程池中多余的線程是如何回收的

也就是說,退出的線程不能互相中斷,我從集合中退出后,中斷了你,你不能中斷我,因為我已經(jīng)退出集合,你只能中斷別人。那么,即使有N個線程同時退出,至少在最后,也會有一條線程,會中斷剩余的阻塞線程。

就像多米諾骨牌一樣,中斷信號就會被傳播下去。

阻塞的C,D中的任意一條被中斷喚醒后,又會重復(fù)step1的動作,周而復(fù)始,直到所有阻塞線程都被中斷,喚醒。

這也是為什么在tryTerminate()里面,傳入false,只需要中斷任意一條空閑線程的原因。

想到這里,再次對Doug Lea心生欽敬(粵語)之情。這設(shè)計得也太妙了叭。

4. 總結(jié)

ThreadPoolExecutor回收工作線程,一條線程getTask()返回null,就會被回收。

分兩種場景。

1) 未調(diào)用shutdown() ,RUNNING狀態(tài)下全部任務(wù)執(zhí)行完成的場景

線程數(shù)量大于corePoolSize,線程超時阻塞,超時喚醒后CAS減少工作線程數(shù),如果CAS成功,返回null,線程回收。否則進入下一次循環(huán)。當工作者線程數(shù)量小于等于corePoolSize,就可以一直阻塞了。

2) 調(diào)用shutdown() ,全部任務(wù)執(zhí)行完成的場景

shutdown() 會向所有線程發(fā)出中斷信號,這時有兩種可能。

2.1)所有線程都在阻塞

中斷喚醒,進入循環(huán),都符合第一個if判斷條件,都返回null,所有線程回收。

2.2)任務(wù)還沒有完全執(zhí)行完

至少會有一條線程被回收。在processWorkerExit(Worker w, boolean completedAbruptly)方法里會調(diào)用tryTerminate(),向任意空閑線程發(fā)出中斷信號。所有被阻塞的線程,最終都會被一個個喚醒,回收。

這一次的分析,昨晚開始寫,寫到一半卡殼,今天早上接著寫,前后花了大概2+2=4個小時寫博客以及1小時思考。

說實話自己還是有點亂,無法一下子理解透徹,也不知道自己理解得對不對。

有沒有用,我也不知道,只能說,加深了對線程池的理解吧(安慰自己),同時也感慨設(shè)計之精妙。

到此這篇關(guān)于Java線程池中多余的線程是如何回收的的文章就介紹到這了,更多相關(guān)Java 多余線程回收內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標簽: Java
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
日韩伦理一区| 欧美成人综合| 美女视频免费精品| 麻豆免费精品视频| 国产高清不卡| 视频一区中文| 青青草国产成人99久久| 色爱综合网欧美| 欧美日韩国产探花| 欧美日韩 国产精品| 91日韩欧美| 丝袜美腿亚洲一区二区图片| 欧美日韩一区二区三区四区在线观看 | 久久激五月天综合精品| 国产成人精品免费视| 99在线观看免费视频精品观看| 国产日韩一区| 久久在线电影| 国产精品网址| 在线亚洲自拍| 精品精品99| 国产一区二区三区亚洲| 国产精品日本欧美一区二区三区| 奇米色欧美一区二区三区| 鲁大师精品99久久久| 久久国产小视频| 国产日韩一区二区三区在线 | 国产一区二区三区黄网站| 亚洲精品91| 久久免费福利| 老色鬼久久亚洲一区二区| 日韩在线视频精品| 欧美日韩精品一区二区三区视频 | 欧美精品不卡| 亚洲精品123区| 久久这里只有| 一二三区精品| 久久香蕉国产| 麻豆久久久久久| 综合亚洲视频| 蜜桃国内精品久久久久软件9| 麻豆国产91在线播放| 亚洲深夜福利在线观看| av在线日韩| 国产极品模特精品一二| 中文在线不卡| 日韩精品一区二区三区免费观看| 欧美片网站免费| 日韩亚洲精品在线| 国产精品二区不卡| 国产毛片精品| 亚洲欧洲专区| 欧美专区一区二区三区| 久久久久久久久丰满| 久久这里只有精品一区二区| 日本欧美一区二区在线观看| 老司机精品久久| 婷婷激情综合| 日韩在线第七页| 精品中文字幕一区二区三区| 日本欧美韩国一区三区| 亚洲一二三区视频| 性欧美长视频| 欧美不卡视频| 97在线精品| 精品一区二区男人吃奶| 欧美日韩中出| 日韩激情综合| 亚州精品视频| 一二三区精品| 中文字幕日本一区| 六月天综合网| 视频一区二区中文字幕| 99riav1国产精品视频| 亚洲性图久久| 欧美日一区二区| 欧美成人综合| 亚洲婷婷在线| 久久久久久久久久久妇女| 国产精品久久久久蜜臀| 国产精品一卡| 欧美激情在线精品一区二区三区| 7m精品国产导航在线| 亚洲精品无吗| 深夜日韩欧美| 欧美日韩一区自拍| 国产免费av国片精品草莓男男 | 日韩中文字幕在线一区| 亚洲伊人影院| 视频一区国产视频| 美国欧美日韩国产在线播放| 激情综合网五月| 欧美成人综合| 久久都是精品| 亚洲综合激情在线| 无码日韩精品一区二区免费| 亚洲人www| 青草国产精品| 国产精品白浆| 精品午夜久久| 丝袜美腿诱惑一区二区三区 | 宅男噜噜噜66国产日韩在线观看| 伊人久久大香线蕉av超碰演员| 91精品电影| 日韩中文字幕区一区有砖一区 | 夜夜嗨一区二区| 在线日韩成人| 欧美综合精品| 久久精品日韩欧美| 国产欧美一区二区三区精品酒店| 久久免费大视频| 国产日韩电影| 精品久久91| 久久不卡日韩美女| 国产精品porn| 欧美日韩国产观看视频| 美腿丝袜在线亚洲一区| 久久久久久色 | 国产亚洲毛片在线| 亚洲性色av| 久久精品国产68国产精品亚洲| 亚洲伦乱视频| 97se综合| 成人国产精品一区二区网站| 在线综合视频| 免费av一区| 亚洲精品少妇| 国产一区二区视频在线看| 久久国产小视频| 亚洲精品自拍| 美腿丝袜亚洲三区| 91精品一区二区三区综合| 亚洲网站视频| 日韩美女精品| 亚洲三级欧美| 视频一区在线视频| 欧美午夜三级| 在线午夜精品| 99热精品久久| 丝袜脚交一区二区| 日本在线不卡视频一二三区| 国产精品亚洲四区在线观看 | 日韩美女国产精品| 精品视频91| 免费国产自久久久久三四区久久| 中文字幕一区二区三区日韩精品| 久久免费影院| 好看不卡的中文字幕| 国产精品视频首页| 桃色一区二区| 青草综合视频| 国产精品久久观看| 亚洲精品一二三区区别| 亚洲精品激情| 久久一区视频| 欧美在线网站| 欧美一区二区三区高清视频 | 久久久精品国产**网站| 午夜影院欧美| 免费在线欧美黄色| 亚洲一区二区毛片| 国产精品sss在线观看av| 亚洲日产国产精品| 亚洲精品中文字幕99999| 精品国产三区在线| 亚洲黄页一区| 美女国产精品久久久| 成人av二区| 欧美韩一区二区| 久久99久久久精品欧美| 国产精品不卡| 青青草伊人久久| 亚洲激情五月| 捆绑调教日本一区二区三区| 91精品视频一区二区| 极品裸体白嫩激情啪啪国产精品| 7m精品国产导航在线| 欧美成人国产| 9999国产精品| 国产精品xxx在线观看| 亚洲深夜福利在线观看| 亚洲午夜精品久久久久久app| 精品99在线| 日韩欧美中文在线观看| 欧美丝袜一区| 日韩成人精品一区二区| 国产亚洲欧美日韩精品一区二区三区| 国产亚洲福利| 亚洲福利国产| 日韩欧美一区二区三区在线视频| 国产精品激情| 免费精品视频最新在线| 99精品视频在线| 色综合五月天| 嫩呦国产一区二区三区av| 日韩精品免费一区二区夜夜嗨| 国产免费成人| 99精品电影| 三级精品视频| 国产精品99一区二区三区|