爲什麼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是一個非公平的鎖,如果競爭激烈的話,可能導致某些線程一直得不到執行

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