本文由https://bbs.csdn.net/topics/390939500和https://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的對象鎖,就不涉及到可重入鎖的問題了。