sleep 跟 wait有什麼區別?

第一:

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 {
public static void main(String args[]){
ThreadB b = new ThreadB();
b.start();
System.out.println("b is start...");
synchronized(b){
   try{
    System.out.println("Waiting for b to complete...");
    b.wait();
    System.out.println("Completed now back to main thread");
   }catch(InterruptedException e){
    
   }
}
System.out.println("total is :"+ b.total);
}

}
/**
* extend the class Thread to realize ThreadB
* notify()與b.wait()相呼應,及計算完事後被喚醒調用
* 因爲一些程序塊是可以分開運行的,但對b的同步操作,必須
* 等到notify()
*/
class ThreadB extends Thread{
int total;
public void run(){
synchronized(this){
   System.out.println("ThreadB is running...");
   for(int i=0; i<100; i++){
    total += i;
    System.out.println("total is " + total);
   }
   notify();
}
}
}


要分析這個程序,首先要理解notify()和wait(),爲什麼在前幾天紀錄線程的時候沒有紀錄這兩個方法呢,因爲這兩個方法本來就不屬於Thread類,而是屬於最底層的object基礎類的,也就是說不光是Thread,每個對象都有notify和wait的功能,爲什麼?因爲他們是用來操縱鎖的,而每個對象都有鎖,鎖是每個對象的基礎,既然鎖是基礎的,那麼操縱鎖的方法當然也是最基礎了.

  再往下看之前呢,首先最好複習一下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所在的線程就可以繼續執行.

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章