你真的知道Java同步鎖何時釋放?

InfoQ 2022-05-14 11:55:09 阅读数:347

真的知道java同步
其他答案:

wait中的線程是不會去競爭對象鎖的。 據我所知,開始由於調用了對象的wait方法,線程處於該對象的等待池中,

而後,只有再去調用對象的notifyAll()(喚醒所有等待池中的線程)或者notify()(隨機喚醒線程,姑且假設喚醒了我們的那個線程),線程會進入該對象的鎖池之中。

鎖池中的對象相互競爭對象鎖,優先級高的線程競爭得到對象鎖的概率高,假若線程沒有競爭到,它還是會在鎖池之中,唯有線程再次調用wait方法,它才會重新回到等待池中。

[](()java多線程什麼時候釋放鎖—wait()、notify()



由於等待一個鎖定線程只有在獲得這把鎖之後,才能恢複運行,所以讓持有鎖的線程在不需要鎖的時候及時釋放鎖是很重要的。在以下情况下,持有鎖的線程會釋放鎖:

  • 執行完同步代碼塊。
  • 在執行同步代碼塊的過程中,遇到异常而導致線程終止。
  • 在執行同步代碼塊的過程中,執行了鎖所屬對象的wait()方法,這個線程會釋放鎖,進行對象的等待池。

除了以上情况外,只要持有鎖的此案吃還沒有執行完同步代碼塊,就不會釋放鎖。因此在以下情况下,線程不會釋放鎖:

  • 在執行同步代碼塊的過程中,執行了Thread.sleep()方法,當前線程放弃CPU,開始睡眠,在睡眠中不會釋放鎖。
  • 在執行同步代碼塊的過程中,執行了Thread.yield()方法,當前線程放弃CPU,但不會釋放鎖。
  • 在執行同步代碼塊的過程中,其他線程執行了當前對象的suspend()方法,當前線程被暫停,但不會釋放鎖。但Thread類的suspend()方法已經被廢弃。

避免死鎖的一個通用的經驗法則是:當幾個線程都要訪問共享資源A、B和C時,保證使每個線程都按照同樣的順序去訪問他們,比如都先訪問A,再訪問B和C。

  • java.lang.Object類中提供了兩個用於線程通信的方法:wait()和notify()。需要注意到是,wait()方法必須放在一個循環中,因為在多線程環境中,共享對象的狀態隨時可能改變。當一個在對象等待池中的線程被喚醒後,並不一定立即恢複運行,等到這個線程獲得了鎖及CPU才能繼續運行,又可能此時對象的狀態已經發生了變化。
  • 調用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()/sleep()的區別

------ 
《一線大廠Java面試題解析+後端開發學習筆記+最新架構講解視頻+實戰項目源碼講義》無償開源 威信搜索公眾號【編程進階路】
 ------------------------------------------------------------------------
版权声明:本文为[InfoQ]所创,转载请带上原文链接,感谢。 https://gsmany.com/2022/134/202205141147349709.html