Java併發(二十一)----wait notify介紹

1、小故事 - 爲什麼需要 wait

  • 由於條件不滿足(沒煙幹不了活啊,等小M把煙送過來),小南不能繼續進行計算

  • 但小南如果一直佔用着鎖,其它人就得一直阻塞,效率太低

  • 於是老王單開了一間休息室(調用 wait 方法),讓小南到休息室(WaitSet)等着去了,但這時鎖釋放開,其它人可以由老王隨機安排進屋

  • 直到小M將煙送來,大叫一聲 [ 你的煙到了 ] (調用 notify 方法)

  • 小南於是可以離開休息室,重新進入競爭鎖的隊列

2、wait notify 原理

  • Owner 線程發現條件不滿足,調用 wait 方法,即可進入 WaitSet 變爲 WAITING 狀態

  • BLOCKED 和 WAITING 的線程都處於阻塞狀態,不佔用 CPU 時間片

  • BLOCKED 線程會在 Owner 線程釋放鎖時喚醒

  • WAITING 線程會在 Owner 線程調用 notify 或 notifyAll 時喚醒,但喚醒後並不意味者立刻獲得鎖,仍需進入 EntryList 重新競爭

3、API 介紹

  • obj.wait() 讓進入 object 監視器的線程到 waitSet 等待

  • obj.notify() 在 object 上正在 waitSet 等待的線程中挑一個喚醒

  • obj.notifyAll() 讓 object 上正在 waitSet 等待的線程全部喚醒

它們都是線程之間進行協作的手段,都屬於 Object 對象的方法。必須獲得此對象的鎖,才能調用這幾個方法,否則會報IllegalMonitorStateException

final static Object obj = new Object();
​
public static void main(String[] args) {
​
    new Thread(() -> {
        synchronized (obj) {
            log.debug("執行....");
            try {
                obj.wait(); // 讓線程在obj上一直等待下去
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            log.debug("其它代碼....");
        }
    }).start();
​
    new Thread(() -> {
        synchronized (obj) {
            log.debug("執行....");
            try {
                obj.wait(); // 讓線程在obj上一直等待下去
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            log.debug("其它代碼....");
        }
    }).start();
​
    // 主線程兩秒後執行
    sleep(2);
    log.debug("喚醒 obj 上其它線程");
    synchronized (obj) {
        obj.notify(); // 喚醒obj上隨機一個線程
        // obj.notifyAll(); // 喚醒obj上所有等待線程
    }
}

notify 的一種結果

20:00:53.096 [Thread-0] c.TestWaitNotify - 執行....
20:00:53.099 [Thread-1] c.TestWaitNotify - 執行....
20:00:55.096 [main] c.TestWaitNotify - 喚醒 obj 上其它線程
20:00:55.096 [Thread-0] c.TestWaitNotify - 其它代碼....

notifyAll 的結果

19:58:15.457 [Thread-0] c.TestWaitNotify - 執行....
19:58:15.460 [Thread-1] c.TestWaitNotify - 執行....
19:58:17.456 [main] c.TestWaitNotify - 喚醒 obj 上其它線程
19:58:17.456 [Thread-1] c.TestWaitNotify - 其它代碼....
19:58:17.456 [Thread-0] c.TestWaitNotify - 其它代碼....

wait() 方法會釋放對象的鎖,進入 WaitSet 等待區,從而讓其他線程就機會獲取對象的鎖。無限制等待,直到 notify 爲止

wait(long n) 有時限的等待, 到 n 毫秒後結束等待,或是被 notify

 

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