Java并發(fā) 線程間的等待與通知
前言:
前面講完了一些并發(fā)編程的原理,現(xiàn)在我們要來(lái)學(xué)習(xí)的是線程之間的協(xié)作。通俗來(lái)說(shuō)就是,當(dāng)前線程在某個(gè)條件下需要等待,不需要使用太多系統(tǒng)資源。在某個(gè)條件下我們需要去喚醒它,分配給它一定的系統(tǒng)資源,讓它繼續(xù)工作。這樣能更好的節(jié)約資源。
一、Object的wait()與notify()
基本概念:
一個(gè)線程因執(zhí)行目標(biāo)動(dòng)作的條件未能滿足而被要求暫停就是wait,而一個(gè)線程滿足執(zhí)行目標(biāo)動(dòng)作的條件之后喚醒被暫停的線程就是notify。
基本模板:
synchronized (obj){ //保護(hù)條件不成立 while(flag){ //暫停當(dāng)前線程 obj.wait(); } //當(dāng)保護(hù)條件成立,即跳出while循環(huán)執(zhí)行目標(biāo)動(dòng)作 doAction(); }
解析wait():Object.wait()的作用是使執(zhí)行線程被暫停,該執(zhí)行線程生命周期就變更為WAITING,這里注意一下,是無(wú)限等待,直到有notify()方法通知該線程喚醒。Object.wait(long timeout)的作用是使執(zhí)行線程超過(guò)一定時(shí)間沒(méi)有被喚醒就自動(dòng)喚醒,也就是超時(shí)等待。Object.wait(long timeout,int naous)是更加精準(zhǔn)的控制時(shí)間的方法,可以控制到毫微秒。這里需要注意的是wait()會(huì)在當(dāng)前線程擁有鎖的時(shí)候才能執(zhí)行該方法并且釋放當(dāng)前線程擁有的鎖,從而讓該線程進(jìn)入等待狀態(tài),其他線程來(lái)嘗試獲取當(dāng)前鎖。也就是需要申請(qǐng)鎖與釋放鎖。
解析notify():Object.notify()方法是喚醒調(diào)用了wait()的線程,只喚醒最多一個(gè)。如果有多個(gè)線程,不一定能喚醒我們所想要的線程。Object.notifyAll()喚醒所有等待的線程。notify方法一定是通知線程先獲取到了鎖才能進(jìn)行通知。通知之后當(dāng)前的通知線程需要釋放鎖,然后由等待線程來(lái)獲取。所以涉及到了一個(gè)申請(qǐng)鎖與釋放鎖的步驟。
wait()與notify()之間存在的三大問(wèn)題:
從上面的解析可以看出,notify()是無(wú)指向性的喚醒,notifyAll()是無(wú)偏差喚醒。所以會(huì)產(chǎn)生下面三個(gè)問(wèn)題
過(guò)早喚醒:假設(shè)當(dāng)前有三組等待(w1,w2,w3)與通知(n1,n2,n3)線程同步在對(duì)象obj上,w1,w2的判斷喚醒條件相同,由線程n1更新條件并喚醒,w3的判斷喚醒條件不同,由n2,n3更新條件并喚醒,這時(shí)如果n1執(zhí)行了喚醒,那么不能執(zhí)行notify,因?yàn)樾枰行褍蓷l線程,只能用notifyAll(),可是用了之后w3的條件未能滿足就被叫醒,就需要一直占用資源的去等待執(zhí)行。
信號(hào)丟失:這個(gè)問(wèn)題主要是程序員編程出現(xiàn)了問(wèn)題,并不是內(nèi)部實(shí)現(xiàn)機(jī)制出現(xiàn)的問(wèn)題。編程時(shí)如果在該使用notifyAll()的地方使用notify()那么只能喚醒一個(gè)線程,從而使其他應(yīng)該喚醒的線程未能喚醒,這就是信號(hào)丟失。如果等待線程在執(zhí)行wait()方法前沒(méi)有先判斷保護(hù)條件是否成立,就會(huì)出現(xiàn)通知線程在該等待線程進(jìn)入臨界區(qū)之前就已經(jīng)更新了相關(guān)共享變量,并且執(zhí)行了notify()方法,但是由于wait()還未能執(zhí)行,且沒(méi)有設(shè)置共享變量的判斷,所以會(huì)執(zhí)行wait()方法,導(dǎo)致線程一直處于等待狀態(tài),丟失了一個(gè)信號(hào)。
欺騙性喚醒:等待線程并不是一定有notify()/notifyAll()才能被喚醒,雖然出現(xiàn)的概率特別低,但是操作系統(tǒng)是允許這種情況發(fā)生的。
上下文切換問(wèn)題:首先wait()至少會(huì)導(dǎo)致線程對(duì)相應(yīng)對(duì)象內(nèi)部鎖的申請(qǐng)與釋放。notify()/notifyAll()時(shí)需要持有相應(yīng)的對(duì)象內(nèi)部鎖并且也會(huì)釋放該鎖,會(huì)出現(xiàn)上下文切換問(wèn)題其實(shí)就是從RUNNABLE狀態(tài)變?yōu)榉荝UNNABLE狀態(tài)會(huì)出現(xiàn)。
針對(duì)問(wèn)題的解決方案:
信號(hào)丟失與欺騙性喚醒問(wèn)題:都可以使用while循環(huán)來(lái)避免,也就是上面的模板中寫(xiě)的那樣。
上下文切換問(wèn)題:在保證程序正確性的情況下使用notify()代替notifyAll(),notify不會(huì)導(dǎo)致過(guò)早喚醒,所以減少了上下文的切換。并且使用了notify之后應(yīng)該盡快釋放相應(yīng)內(nèi)部鎖,從而讓wait()能夠更快的申請(qǐng)到鎖。
過(guò)早喚醒:使用java.util.concurrent.locks.Condition中的await與signal。
PS:由于Object中的wait與notify使用的是native方法,即C++編寫(xiě),這里不做源碼解析。
二、Condition中的await()與signal()
這個(gè)方法相應(yīng)的改變了上面所說(shuō)的無(wú)指向性的問(wèn)題,每個(gè)Condition內(nèi)部都會(huì)維護(hù)一個(gè)隊(duì)列,從而讓我們對(duì)線程之間的操作更加靈活。下面通過(guò)分析源碼讓我們了解一下內(nèi)部機(jī)制。Condition是個(gè)接口,真正的實(shí)現(xiàn)是AbstractQueuedSynchronizer中的內(nèi)部類ConditionObject。
基本屬性:
public class ConditionObject implements Condition, java.io.Serializable { private static final long serialVersionUID = 1173984872572414699L; /** First node of condition queue. */ private transient Node firstWaiter; /** Last node of condition queue. */ private transient Node lastWaiter; }
從基本屬性中可看出維護(hù)的是雙端隊(duì)列。
await()方法解析:
public class ConditionObject implements Condition, java.io.Serializable { public final void await() throws InterruptedException { // 1. 判斷線程是否中斷 if(Thread.interrupted()){ throw new InterruptedException(); } // 2. 將線程封裝成一個(gè) Node 放到 Condition Queue 里面 Node node = addConditionWaiter(); // 3. 釋放當(dāng)前線程所獲取的所有的鎖 (PS: 調(diào)用 await 方法時(shí), 當(dāng)前線程是必須已經(jīng)獲取了獨(dú)占的鎖) int savedState = fullyRelease(node); int interruptMode = 0; // 4. 判斷當(dāng)前線程是否在 Sync Queue 里面(這里 Node 從 Condtion Queue 里面轉(zhuǎn)移到 Sync Queue 里面有兩種可能 //(1) 其他線程調(diào)用 signal 進(jìn)行轉(zhuǎn)移 (2) 當(dāng)前線程被中斷而進(jìn)行Node的轉(zhuǎn)移(就在checkInterruptWhileWaiting里面進(jìn)行轉(zhuǎn)移)) while(!isOnSyncQueue(node)){ // 5. 當(dāng)前線程沒(méi)在 Sync Queue 里面, 則進(jìn)行 block LockSupport.park(this); // 6. 判斷此次線程的喚醒是否因?yàn)榫€程被中斷, 若是被中斷, 則會(huì)在checkInterruptWhileWaiting的transferAfterCancelledWait 進(jìn)行節(jié)點(diǎn)的轉(zhuǎn)移; if((interruptMode = checkInterruptWhileWaiting(node)) != 0){ // 說(shuō)明此是通過(guò)線程中斷的方式進(jìn)行喚醒, 并且已經(jīng)進(jìn)行了 node 的轉(zhuǎn)移, 轉(zhuǎn)移到 Sync Queue 里面 break; } } // 7. 調(diào)用 acquireQueued在 Sync Queue 里面進(jìn)行獨(dú)占鎖的獲取, 返回值表明在獲取的過(guò)程中有沒(méi)有被中斷過(guò) if(acquireQueued(node, savedState) && interruptMode != THROW_IE){ interruptMode = REINTERRUPT; } // 8. 通過(guò) "node.nextWaiter != null" 判斷 線程的喚醒是中斷還是 signal。 //因?yàn)橥ㄟ^(guò)中斷喚醒的話, 此刻代表線程的 Node 在 Condition Queue 與 Sync Queue 里面都會(huì)存在 if(node.nextWaiter != null){ // 9. 進(jìn)行 cancelled 節(jié)點(diǎn)的清除 unlinkCancelledWaiters(); } // 10. "interruptMode != 0" 代表通過(guò)中斷的方式喚醒線程 if(interruptMode != 0){ // 11. 根據(jù) interruptMode 的類型決定是拋出異常, 還是自己再中斷一下 reportInterruptAfterWait(interruptMode); } } }
上面源代碼可看出Condition內(nèi)部維護(hù)的隊(duì)列是一個(gè)等待隊(duì)列,當(dāng)需要調(diào)用signal()方法時(shí)就會(huì)讓當(dāng)前線程節(jié)點(diǎn)從Condition queue轉(zhuǎn)到Sync queue隊(duì)列中去競(jìng)爭(zhēng)鎖從而喚醒。
signal()源碼解析:
public class ConditionObject implements Condition, java.io.Serializable { public final void signal() { if (!isHeldExclusively()) throw new IllegalMonitorStateException(); Node first = firstWaiter; if (first != null) doSignal(first); } private void doSignal(Node first) { do { //傳入的鏈表下一個(gè)節(jié)點(diǎn)為空,則尾節(jié)點(diǎn)置空 if ( (firstWaiter = first.nextWaiter) == null) lastWaiter = null; //當(dāng)前節(jié)點(diǎn)的下一個(gè)節(jié)點(diǎn)為空 first.nextWaiter = null; //如果成功將node從condition queue轉(zhuǎn)換到sync queue,則退出循環(huán),節(jié)點(diǎn)為空了也退出循環(huán)。否則就接著在隊(duì)列中找尋節(jié)點(diǎn)進(jìn)行喚醒 } while (!transferForSignal(first) && (first = firstWaiter) != null); } }
signal()會(huì)使等待隊(duì)列中的一個(gè)任意線程被喚醒,signalAll()則是喚醒該隊(duì)列中的所有線程。這樣通過(guò)不同隊(duì)列維護(hù)不同線程,就可以達(dá)到指向性的功能??梢韵蛇^(guò)早喚醒帶來(lái)的資源損耗。注意的是在使用signal()方法前需要獲取鎖,即lock(),而后需要盡快unlock(),這樣可以避免上下文切換的損耗。
總結(jié):
面向?qū)ο蟮氖澜缰?,一個(gè)類往往需要借助其他的類來(lái)一起完成計(jì)算,同樣線程的世界也是,多個(gè)線程可以同時(shí)完成一個(gè)任務(wù),通過(guò)喚醒與等待,能更好的操作線程,從而讓線程在需要使用資源的時(shí)候分配資源給它,而不使用資源的時(shí)候就可以將資源讓給其他線程操作。關(guān)于Condition中提到的Sync queue可參考Java并發(fā) 結(jié)合源碼分析AQS原理來(lái)看內(nèi)部維護(hù)的隊(duì)列是如何獲取鎖的。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java日常練習(xí)題,每天進(jìn)步一點(diǎn)點(diǎn)(52)
下面小編就為大家?guī)?lái)一篇Java基礎(chǔ)的幾道練習(xí)題(分享)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧,希望可以幫到你2021-08-08java常用工具類 Random隨機(jī)數(shù)、MD5加密工具類
這篇文章主要為大家詳細(xì)介紹了Java常用工具類,Random隨機(jī)數(shù)工具類、MD5加密工具類,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-05-05Mybatis中特殊SQL的執(zhí)行的實(shí)現(xiàn)示例
本文主要介紹了Mybatis中特殊SQL的執(zhí)行的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-07-07Java簡(jiǎn)單計(jì)算器的實(shí)現(xiàn)
這篇文章主要為大家詳細(xì)介紹了Java簡(jiǎn)單計(jì)算器的實(shí)現(xiàn),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-12-12Java實(shí)現(xiàn)redis分布式鎖的三種方式
本文主要介紹了Java實(shí)現(xiàn)redis分布式鎖的三種方式,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08Java?通過(guò)手寫(xiě)分布式雪花SnowFlake生成ID方法詳解
SnowFlake是twitter公司內(nèi)部分布式項(xiàng)目采用的ID生成算法,開(kāi)源后廣受國(guó)內(nèi)大廠的好評(píng)。由這種算法生成的ID,我們就叫做SnowFlakeID,下面我們來(lái)詳細(xì)看看2022-04-04