1. 程式人生 > >為什麼wait,notify和notifyAll要與synchronized一起使用?

為什麼wait,notify和notifyAll要與synchronized一起使用?

Object.wait(),Object.notify(),Object.notifyAll()都是Object的方法,換句話說,就是每個類裡面都有這些方法。

  • Object.wait():釋放當前物件鎖,並進入阻塞佇列
  • Object.notify():喚醒當前物件阻塞佇列裡的任一執行緒(並不保證喚醒哪一個)
  • Object.notifyAll():喚醒當前物件阻塞佇列裡的所有執行緒

為什麼這三個方法要與synchronized一起使用呢?解釋這個問題之前,我們先要了解幾個知識點

  • 每一個物件都有一個與之對應的監視器
  • 每一個監視器裡面都有一個該物件的鎖和一個等待佇列和一個同步佇列

wait()方法的語義有兩個,一是釋放當前物件鎖,另一個是進入阻塞佇列,可以看到,這些操作都是與監視器相關的,當然要指定一個監視器才能完成這個操作了

notify()方法也是一樣的,用來喚醒一個執行緒,你要去喚醒,首先你得知道他在哪兒,所以必須先找到該物件,也就是獲取該物件的鎖,當獲取到該物件的鎖之後,才能去該物件的對應的等待佇列去喚醒一個執行緒。值得注意的是,只有當執行喚醒工作的執行緒離開同步塊,即釋放鎖之後,被喚醒執行緒才能去競爭鎖。

notifyAll()方法和notify()一樣,只不過是喚醒等待佇列中的所有執行緒

wait()而導致阻塞的執行緒是放在阻塞佇列中的,因競爭失敗導致的阻塞是放在同步佇列中的,notify()/notifyAll()實質上是把阻塞佇列中的執行緒放到同步佇列中

為了便於理解,你可以把執行緒想象成一個個列車

,物件想象成車站,每一個車站每一次能跑一班車,這樣理解起來就比較容易了。

值得提的一點是,synchronized是一個非公平的鎖,如果競爭激烈的話,可能導致某些執行緒一直得不到執行