sleep( ) 和 wait( ) 的這 5 個區別

參考:Java技術棧公衆號文章

sleep(休眠) 和 wait(等待) 方法是 Java 多線程中常用的兩個方法,它們有什麼區別及一些該注意的地方有哪些呢?下面給大家一一分解。

區別1:使用限制

使用 sleep 方法可以讓讓當前線程休眠,時間一到當前線程繼續往下執行,在任何地方都能使用,但需要捕獲 InterruptedException 異常。


 

try {
    Thread.sleep(3000L);
} catch (InterruptedException e) {
    e.printStackTrace();
}

而使用 wait 方法則必須放在 synchronized 塊裏面,同樣需要捕獲 InterruptedException 異常,並且需要獲取對象的鎖。


 

synchronized (lock){
    try {
        lock.wait();
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
}

而且 wait 還需要額外的方法 notify/ notifyAll 進行喚醒,它們同樣需要放在 synchronized 塊裏面,且獲取對象的鎖。。


 

synchronized (lock) {
    // 隨機喚醒
    lock.notify();

    // 喚醒全部
    lock.notifyAll();
}

當然也可以使用帶時間的 wait(long millis) 方法,時間一到,無需其他線程喚醒,也會重新競爭獲取對象的鎖繼續執行。

區別2:使用場景

sleep 一般用於當前線程休眠,或者輪循暫停操作,wait 則多用於多線程之間的通信。

區別3:所屬類

sleep 是 Thread 類的靜態本地方法,wait 則是 Object 類的本地方法。

java.lang.Thread#sleep

public static native void sleep(long millis) throws InterruptedException;

java.lang.Object#wait

public final native void wait(long timeout) throws InterruptedException;

爲什麼要這樣設計呢?

因爲 sleep 是讓當前線程休眠,不涉及到對象類,也不需要獲得對象的鎖,所以是線程類的方法。wait 是讓獲得對象鎖的線程實現等待,前提是要楚獲得對象的鎖,所以是類的方法。

區別4:釋放鎖

Object lock = new Object();
synchronized (lock) {
    try {
        lock.wait(3000L);
        Thread.sleep(2000L);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
}

如上代碼所示,wait 可以釋放當前線程對 lock 對象鎖的持有,而 sleep 則不會。

區別5:線程切換

sleep 會讓出 CPU 執行時間且強制上下文切換,而 wait 則不一定,wait 後可能還是有機會重新競爭到鎖繼續執行的。

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