java 多线程 notify_Java多线程8:wait()和notify()/notifyAll()
輪詢
線程本身是操作系統(tǒng)中獨(dú)立的個體,但是線程與線程之間不是獨(dú)立的個體,因?yàn)樗鼈儽舜酥g要相互通信和協(xié)作。
想像一個場景,A線程做int型變量i的累加操作,B線程等待i到了10000就打印出i,怎么處理?一個辦法就是,B線程while(i == 10000),這樣兩個線程之間就有了通信,B線程不斷通過輪訓(xùn)來檢測i == 10000這個條件。
這樣可以實(shí)現(xiàn)我們的需求,但是也帶來了問題:CPU把資源浪費(fèi)了B線程的輪詢操作上,因?yàn)閣hile操作并不釋放CPU資源,導(dǎo)致了CPU會一直在這個線程中做判斷操作。如果可以把這些輪詢的時間釋放出來,給別的線程用,就好了。
wait/notify
在Object對象中有三個方法wait()、notify()、notifyAll(),既然是Object中的方法,那每個對象自然都是有的。如果不接觸多線程的話,這兩個方法是不太常見的。下面看一下前兩個方法:
1、wait()
wait()的作用是使當(dāng)前執(zhí)行代碼的線程進(jìn)行等待,將當(dāng)前線程置入"預(yù)執(zhí)行隊(duì)列"中,并且wait()所在的代碼處停止執(zhí)行,直到接到通知或被中斷。在調(diào)用wait()之前,線程必須獲得該對象的鎖,因此只能在同步方法/同步代碼塊中調(diào)用wait()方法。
2、notify()
notify()的作用是,如果有多個線程等待,那么線程規(guī)劃器隨機(jī)挑選出一個wait的線程,對其發(fā)出通知notify(),并使它等待獲取該對象的對象鎖。注意"等待獲取該對象的對象鎖",這意味著,即使收到了通知,wait的線程也不會馬上獲取對象鎖,必須等待notify()方法的線程釋放鎖才可以。和wait()一樣,notify()也要在同步方法/同步代碼塊中調(diào)用。
總結(jié)起來就是,wait()使線程停止運(yùn)行,notify()使停止運(yùn)行的線程繼續(xù)運(yùn)行。
wait()/notify()使用示例
看一段代碼:
寫個main函數(shù),同樣的Thread.sleep(3000)也是為了保證mt0先運(yùn)行,這樣才能看到wait()和notify()的效果:
看一下運(yùn)行結(jié)果:
第一行和第二行之間的time減一下很明顯就是3s,說明wait()之后代碼一直暫停,notify()之后代碼才開始運(yùn)行。
wait()方法可以使調(diào)用該線程的方法釋放共享資源的鎖,然后從運(yùn)行狀態(tài)退出,進(jìn)入等待隊(duì)列,直到再次被喚醒。
notify()方法可以隨機(jī)喚醒等待隊(duì)列中等待同一共享資源的一個線程,并使得該線程退出等待狀態(tài),進(jìn)入可運(yùn)行狀態(tài)
notifyAll()方法可以使所有正在等待隊(duì)列中等待同一共享資源的全部線程從等待狀態(tài)退出,進(jìn)入可運(yùn)行狀態(tài)
最后,如果wait()方法和notify()/notifyAll()方法不在同步方法/同步代碼塊中被調(diào)用,那么虛擬機(jī)會拋出java.lang.IllegalMonitorStateException,注意一下。
wait()釋放鎖以及notify()不釋放鎖
多線程的學(xué)習(xí)中,任何地方都要關(guān)注"鎖",wait()和notify()也是這樣。wait()方法是釋放鎖的,寫一個例子來證明一下:
如果wait()方法不釋放鎖,那么Thread-1根本不會進(jìn)入同步代碼塊打印的,所以,證明完畢。
接下來證明一下notify()方法不釋放鎖的結(jié)論:
寫兩個線程分別調(diào)用2個方法:
看一下運(yùn)行結(jié)果:
如果notify()方法釋放鎖,那么在Thread-1調(diào)用notify()方法后Thread.sleep(5000)必定應(yīng)該有其他線程可以進(jìn)入同步代碼塊了,但是實(shí)際上沒有,必須等到Thread-1把代碼執(zhí)行完。所以,證明完畢。
interrupt()打斷wait()
之前有說過,interrupt()方法的作用不是中斷線程,而是在線程阻塞的時候給線程一個中斷標(biāo)識,表示該線程中斷。wait()就是"阻塞的一種場景",看一下用interrupt()打斷wait()的例子:
notifyAll()喚醒所有線程
利用Object對象的notifyAll()方法可以喚醒處于同一監(jiān)視器下的所有處于wait的線程,舉個例子證明一下:
寫兩個線程,一個調(diào)用testMethod(Object lock)的線程,一個notifyAll()線程:
main函數(shù)開三個wait線程,用一個notifyAll的線程去喚醒:
當(dāng)然,喚醒的順序不重要,因?yàn)閚otifyAll()把處于同一資源下wait的線程全部喚醒,至于喚醒的順序,就和線程啟動的順序一樣,是虛擬機(jī)隨機(jī)的。
總結(jié)
以上是生活随笔為你收集整理的java 多线程 notify_Java多线程8:wait()和notify()/notifyAll()的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: android开发 Gradle多渠道
- 下一篇: 关于《Java编程思想》的简单纠正