Java可重入鎖如何避免死鎖


  本文由https://bbs.csdn.net/topics/390939500https://zhidao.baidu.com/question/1946051090515119908.html啟發而來。

 

  看到一個問題,Java的可重入鎖為什么可以防止死鎖呢?網上看了看資料,雖然有答案說出了正確答案,但是分析的不夠詳細,對初學者不夠友好。這里我再做一個更清晰的分析。

  首先是示例代碼:

 1 public class Widget {
 2     public synchronized void doSomething(){
 3         // do something
 4     }
 5 }
 6 public class LoggingWidget extends Widget {
 7     public synchronized void doSomething() {
 8         super.doSomething();
 9     }
10 }

   這是《java並發編程實例》一書中的例子,並且書中說:“如果synchronized 不是可重入鎖,那么LoggingWidget 的super.dosomething();無法獲得Widget對象的鎖,因為會死鎖。”

  

  乍一看好像不是這么回事,就算synchronized 不是可重入鎖,可是synchronized 關鍵字一個在父類Widget 的方法上,另一個在子類LoggingWidget 的方法上,怎么會有死鎖產生呢。

  這里其實牽涉到了Java的重寫。我們看子類LoggingWidget 的doSomething方法,重寫了父類Widget 的doSomething方法,但是子類對象如果要調用父類的doSomething方法,那么就需要用到super關鍵字了。因為實例方法的調用是Java虛擬機在運行時動態綁定的,子類LoggingWidget 的對象調用doSomething方法,一定是綁定到子類自身的doSomething方法,必須用super關鍵字告訴虛擬機,這里要調用的是父類的doSomething方法。

  實際上,如果我們分析運行時的LoggingWidget 類,那我們看到的應該是這樣子的(這里只是為了分析,真實情況肯定和下面的例子不同):

1 public class LoggingWidget extends Widget {
2     public synchronized void Widget.doSomething() {
3         // do something
4     }   // 父類的doSomething方法
5 
6     public synchronized void doSomething() {
7         super.doSomething();
8     }
9 }

 

  子類對象,其實是持有父類Widget 的doSomething方法的,只需要使用super關鍵字告訴虛擬機要運行的是父類的doSomething方法,虛擬機會去調用子類對象中的父類Widget 的doSomething方法的。所以,super關鍵字並沒有新建一個父類的對象,比如說widget,然后再去調用widget.doSomething方法,實際上調用父類doSomething方法的還是我們的子類對象。

  那么這樣就很好理解了,如果一個線程有子類對象的引用loggingWidget,然后調用loggingWidget.doSomething方法的時候,會請求子類對象loggingWidget 的對象鎖;又因為loggingWidget 的doSomething方法中調用的父類的doSomething方法,實際上還是要請求子類對象loggingWidget 的對象鎖,那么如果synchronized 關鍵字不是個可重入鎖的話,就會在子類對象持有的父類doSomething方法上產生死鎖了。正因為synchronized 關鍵字的可重入鎖,當前線程因為已經持有了子類對象loggingWidget 的對象鎖,后面再遇到請求loggingWidget 的對象鎖就可以暢通無阻地執行同步方法了。

 

  更進一步,將上面的示例代碼改寫一下,那么就算synchronized 不是可重入鎖,也不會產生死鎖的問題了。代碼如下:

 1 public class Widget {
 2     public synchronized void doSomething(){
 3         // do something
 4     }
 5 }
 6 public class LoggingWidget extends Widget {
 7     public synchronized void doSomething() {
 8         Widget widget = new Widget();
 9         widget.doSomething();
10     }
11 }

 

  在子類的doSomething方法中,直接新建了一個父類的對象widget,然后用這個父類對象來調用父類的doSomething方法,實際上請求的是這個父類對象widget的對象鎖,就不涉及到可重入鎖的問題了。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM