同步synchronized方法和代碼塊

 

Java中,等待機制與鎖機制是密切關聯的。例如:

synchronized(obj) {
    while(!condition) {
        obj.wait();
    }
    obj.doSomething();
}

當線程A獲得了obj鎖後,發現條件condition不滿足,無法繼續下一處理,於是線程A就wait()。

在另一線程B中,如果B更改了某些條件,使得線程A的condition條件滿足了,就可以喚醒線程A:

synchronized(obj) {
    condition = true;
    obj.notify();
}

需要注意的概念是:

# 調用obj的wait(), notify()方法前,必須獲得obj鎖,也就是必須寫在synchronized(obj) {...} 代碼段內。

# 調用obj.wait()後,線程A就釋放了obj的鎖,否則線程B無法獲得obj鎖,也就無法在synchronized(obj) {...} 代碼段內喚醒A。

# 當obj.wait()方法返回後,線程A需要再次獲得obj鎖,才能繼續執行。

# 如果A1,A2,A3都在obj.wait(),則B調用obj.notify()只能喚醒A1,A2,A3中的一個(具體哪一個由JVM決定)。

# obj.notifyAll()則能全部喚醒A1,A2,A3,但是要繼續執行obj.wait()的下一條語句,必須獲得obj鎖,因此,A1,A2,A3只有一個有機會獲得鎖繼續執行,例如A1,其餘的需要等待A1釋放obj鎖之後才能繼續執行。

# 當B調用obj.notify/notifyAll的時候,B正持有obj鎖,因此,A1,A2,A3雖被喚醒,但是仍無法獲得obj鎖。直到B退出synchronized塊,釋放obj鎖後,A1,A2,A3中的一個纔有機會獲得鎖繼續執行。

 

wait(),wait(long),notify(),notifyAll()等方法是當前類的實例方法,
   
        wait()是使持有對象鎖的線程釋放鎖;
        wait(long)是使持有對象鎖的線程釋放鎖時間爲long(毫秒)後,再次獲得鎖,wait()和wait(0)等價;
        notify()是喚醒一個正在等待該對象鎖的線程,如果等待的線程不止一個,那麼被喚醒的線程由jvm確定;
        notifyAll是喚醒所有正在等待該對象鎖的線程. 應該優先使用notifyAll()方法,因爲喚醒所有線程比喚醒一個線程更容易讓jvm找到最適合被喚醒的線程.

1.有synchronized的地方不一定有wait,notify

2.有wait,notify的地方必有synchronized.這是因爲wait和notify不是屬於線程類,而是每一個對象都具有的方法,而且,這兩個方法都和對象鎖有關,有鎖的地方,必有synchronized。

另外,請注意一點:如果要把notify和wait方法放在一起用的話,必須先調用notify後調用wait,因爲如果調用完wait,該線程就已經不是current thread了

發佈了32 篇原創文章 · 獲贊 1 · 訪問量 9萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章