concurrent: wai notify notifyAll
轉(zhuǎn)載自:http://sishuok.com/forum/blogPost/list/3674.html
?
通常,多線程之間需要協(xié)調(diào)工作。例如,瀏覽器的一個(gè)顯示圖片的線程displayThread想要執(zhí)行顯示圖片的任務(wù),必須等待下載線程 downloadThread將該圖片下載完畢。如果圖片還沒(méi)有下載完,displayThread可以暫停,當(dāng)downloadThread完成了任務(wù) 后,再通知displayThread“圖片準(zhǔn)備完畢,可以顯示了”,這時(shí),displayThread繼續(xù)執(zhí)行。
以上邏輯簡(jiǎn)單的說(shuō)就是:如果條件不滿足,則等待。當(dāng)條件滿足時(shí),等待該條件的線程將被喚醒。在Java中,這個(gè)機(jī)制的實(shí)現(xiàn)依賴于wait/notify。等待機(jī)制與鎖機(jī)制是密切關(guān)聯(lián)的。例如:
synchronized(obj) {while(!condition) {obj.wait();}obj.doSomething();}??
當(dāng)線程A獲得了obj鎖后,發(fā)現(xiàn)條件condition不滿足,無(wú)法繼續(xù)下一處理,于是線程A就wait()。
在另一線程B中,如果B更改了某些條件,使得線程A的condition條件滿足了,就可以喚醒線程A:
synchronized(obj) {condition = true;obj.notify();}
需要注意的概念是:
◆調(diào)用obj的wait(), notify()方法前,必須獲得obj鎖,也就是必須寫(xiě)在synchronized(obj) {...} 代碼段內(nèi)。
◆調(diào)用obj.wait()后,線程A就釋放了obj的鎖,否則線程B無(wú)法獲得obj鎖,也就無(wú)法在synchronized(obj) {...} 代碼段內(nèi)喚醒A。
◆當(dāng)obj.wait()方法返回后,線程A需要再次獲得obj鎖,才能繼續(xù)執(zhí)行。
◆如果A1,A2,A3都在obj.wait(),則B調(diào)用obj.notify()只能喚醒A1,A2,A3中的一個(gè)(具體哪一個(gè)由JVM決定)。
◆obj.notifyAll()則能全部喚醒A1,A2,A3,但是要繼續(xù)執(zhí)行obj.wait()的下一條語(yǔ)句,必須獲得obj鎖,因此,A1,A2,A3只有一個(gè)有機(jī)會(huì)獲得鎖繼續(xù)執(zhí)行,例如A1,其余的需要等待A1釋放obj鎖之后才能繼續(xù)執(zhí)行。
◆當(dāng)B調(diào)用obj.notify/notifyAll的時(shí)候,B正持有obj鎖,因此,A1,A2,A3雖被喚醒,但是仍無(wú)法獲得obj鎖。直到B退出synchronized塊,釋放obj鎖后,A1,A2,A3中的一個(gè)才有機(jī)會(huì)獲得鎖繼續(xù)執(zhí)行
public class TestWaitNotify {// 鎖Byte lock = 0;// 條件,當(dāng)條件為真,AThread才能往下執(zhí)行,否則會(huì)等待,直到有線程將條件改為真。Boolean condition = false;//內(nèi)部類:等待線程 class AThread extends Thread {public void methodA() {if (condition) {System.out.println("方法methodA執(zhí)行");} else {try {/** 這里要特別注意,* (1)只能在同步控制方法或同步塊中調(diào)用wait()、notify()和notifyAll()。如果在非同步的方法里調(diào)用這些方法,在運(yùn)行時(shí)會(huì)拋出IllegalMonitorStateException異常。* (2)synchronized的鎖與notity的調(diào)用對(duì)象必須一致,否則也會(huì)拋出IllegalMonitorStateException異常;* (3)下面的代碼也會(huì)使拋出異常:* synchronized(condtion){condition=true;condition.wait();* 因?yàn)閏ondition改變以后,synchronized中的對(duì)象與調(diào)用wait方法的對(duì)象已經(jīng)不同。* 所以condtion與lock要分開(kāi),因?yàn)閏ondtion會(huì)改變* */synchronized (lock) {System.out.println("方法methodA等待");lock.wait();System.out.println("方法methodA等待結(jié)束");}} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();}}}@Overridepublic void run() {methodA();}}//內(nèi)部類 notify線程class BThread extends Thread {@Overridepublic void run() {synchronized (lock) {System.out.println("方法methodB執(zhí)行");condition = true;lock.notifyAll();System.out.println("方法methodB喚醒");}}}public static void main(String[] args) {TestWaitNotify t = new TestWaitNotify();TestWaitNotify.AThread aThread = t.new AThread();aThread.start();TestWaitNotify.BThread bThread = t.new BThread();bThread.start();} }
?
總結(jié)
以上是生活随笔為你收集整理的concurrent: wai notify notifyAll的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: tomcat quartz 被触
- 下一篇: 日志管理(一):slf4j原理简单介绍