第一: sleep是Thread對象裏面的方法; wait是Object對象裏面的方法;
第二: 當一個synchorized成員sleep的時候並不會釋放掉synchorized; 但是當一個synchorized成員wait的時候是會釋放掉synchorized的;
第三: 一個對象在sleep的時候是有時間設定限制的,除非在sleep設定時間內被中途打斷,就會拋出InterruptedException異常; 一個對象在wait的時候沒有時間限制,會讓對象一直暫停下去,當有其他對象去調用他的notify或者調用全局的notifyAll的時候纔會開始繼續執行。
這兩者的施加者是有本質區別的,從而對應不同的應用環境. sleep()是讓某個線程暫停運行一段時間,其控制範圍是由當前線程決定,也就是說,在線程裏面決定.好比如說,我要做的事情是 "點火->燒水->煮麪",而當我點完火之後我不立即燒水,我要休息一段時間再燒.對於運行的主動權是由我的流程來控制.
而wait(),首先,這是由某個確定的對象來調用的,將這個對象理解成一個傳話的人,當這個人在某個線程裏面說"暫停!",也是 thisOBJ.wait(),這裏的暫停是阻塞,還是"點火->燒水->煮飯",thisOBJ就好比一個監督我的人站在我旁邊,本來該線 程應該執行1後執行2,再執行3,而在2處被那個對象喊暫停,那麼我就會一直等在這裏而不執行3,但正個流程並沒有結束,我一直想去煮飯,但還沒被允許, 直到那個對象在某個地方說"通知暫停的線程啓動!",也就是thisOBJ.notify()的時候,那麼我就可以煮飯了,這個被暫停的線程就會從暫停處 繼續執行.
其實兩者都可以讓線程暫停一段時間,但是本質的區別是一個線程的運行狀態控制,一個是線程之間的通訊的問題
在java.lang.Thread類中,提供了sleep(), 而java.lang.Object類中提供了wait(), notify()和notifyAll()方法來操作線程 sleep()可以將一個線程睡眠,參數可以指定一個時間。 而wait()可以將一個線程掛起,直到超時或者該線程被喚醒。 wait有兩種形式wait()和wait(milliseconds). sleep和wait的區別有: 1,這兩個方法來自不同的類分別是Thread和Object 2,最主要是sleep方法沒有釋放鎖,而wait方法釋放了鎖,使得其他線程可以使用同步控制塊或者方法。 3,wait,notify和notifyAll只能在同步控制方法或者同步控制塊裏面使用,而sleep可以在 任何地方使用 synchronized(x){ x.notify() //或者wait() } 4,sleep必須捕獲異常,而wait,notify和notifyAll不需要捕獲異常
//分析這段程序,並解釋一下,着重講講synchronized、wait(),notify 謝謝!
public class WaitAndNotify {
}
再往下看之前呢,首先最好複習一下Think in Java的14.3.1中第3部分內容:等待和通知,也就是wait()和notify了.
按照Think in Java中的解釋:\"wait()允許我們將線程置入“睡眠”狀態,同時又“積極”地等待條件發生改變.而且只有在一個notify()或notifyAll()發生變化的時候,線程纔會被喚醒,並檢查條件是否有變.\"
我們來解釋一下這句話. \"wait()允許我們將線程置入“睡眠”狀態\",也就是說,wait也是讓當前線程阻塞的,這一點和sleep或者suspend是相同的.那和sleep,suspend有什麼區別呢?
區別在於\"(wait)同時又“積極”地等待條件發生改變\",這一點很關鍵,sleep和suspend無法做到.因爲我們有時候需要通過同步(synchronized)的幫助來防止線程之間的衝突,而一旦使用同步,就要鎖定對象,也就是獲取對象鎖,其它要使用該對象鎖的線程都只能排隊等着,等到同步方法或者同步塊裏的程序全部運行完纔有機會.在同步方法和同步塊中,無論sleep()還是suspend()都不可能自己被調用的時候解除鎖定,他們都霸佔着正在使用的對象鎖不放. 而wait卻可以,它可以讓同步方法或者同步塊暫時放棄對象鎖,而將它暫時讓給其它需要對象鎖的人(這裏應該是程序塊,或線程)用,這意味着可在執行wait()期間調用線程對象中的其他同步方法!在其它情況下(sleep啊,suspend啊),這是不可能的. 但是注意我前面說的,只是暫時放棄對象鎖,暫時給其它線程使用,我wait所在的線程還是要把這個對象鎖收回來的呀.wait什麼?就是wait別人用完了還給我啊! 好,那怎麼把對象鎖收回來呢? 第一種方法,限定借出去的時間.在wait()中設置參數,比如wait(1000),以毫秒爲單位,就表明我只借出去1秒中,一秒鐘之後,我自動收回. 第二種方法,讓借出去的人通知我,他用完了,要還給我了.這時,我馬上就收回來.哎,假如我設了1小時之後收回,別人只用了半小時就完了,那怎麼辦呢?靠!當然用完了就收回了,還管我設的是多長時間啊.
那麼別人怎麼通知我呢?相信大家都可以想到了,notify(),這就是最後一句話\"而且只有在一個notify()或notifyAll()發生變化的時候,線程纔會被喚醒\"的意思了. 因此,我們可將一個wait()和notify()置入任何同步方法或同步塊內部,無論在那個類裏是否準備進行涉及線程的處理。而且實際上,我們也只能在同步方法或者同步塊裏面調用wait()和notify().
這個時候我們來解釋上面的程序,簡直是易如反掌了.
synchronized(b){...};的意思是定義一個同步塊,使用b作爲資源鎖。b.wait();的意思是臨時釋放鎖,並阻塞當前線程,好讓其他使用同一把鎖的線程有機會執行,在這裏要用同一把鎖的就是b線程本身.這個線程在執行到一定地方後用notify()通知wait的線程,鎖已經用完,待notify()所在的同步塊運行完之後,wait所在的線程就可以繼續執行. |
sleep 跟 wait有什麼區別?
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.