java代码如何避免死锁,Java可重入锁如何避免死锁
看到一個(gè)問(wèn)題,Java的可重入鎖為什么可以防止死鎖呢?網(wǎng)上看了看資料,雖然有答案說(shuō)出了正確答案,但是分析的不夠詳細(xì),對(duì)初學(xué)者不夠友好。這里我再做一個(gè)更清晰的分析。
首先是示例代碼:
1 public classWidget {2 public synchronized voiddoSomething(){3 //do something
4 }5 }6 public class LoggingWidget extendsWidget {7 public synchronized voiddoSomething() {8 super.doSomething();9 }10 }
這是《java并發(fā)編程實(shí)例》一書(shū)中的例子,并且書(shū)中說(shuō):“如果synchronized 不是可重入鎖,那么LoggingWidget 的super.dosomething();無(wú)法獲得Widget對(duì)象的鎖,因?yàn)闀?huì)死鎖。”
乍一看好像不是這么回事,就算synchronized不是可重入鎖,可是synchronized關(guān)鍵字一個(gè)在父類(lèi)Widget 的方法上,另一個(gè)在子類(lèi)LoggingWidget 的方法上,怎么會(huì)有死鎖產(chǎn)生呢。
這里其實(shí)牽涉到了Java的重寫(xiě)。我們看子類(lèi)LoggingWidget 的doSomething方法,重寫(xiě)了父類(lèi)Widget 的doSomething方法,但是子類(lèi)對(duì)象如果要調(diào)用父類(lèi)的doSomething方法,那么就需要用到super關(guān)鍵字了。因?yàn)閷?shí)例方法的調(diào)用是Java虛擬機(jī)在運(yùn)行時(shí)動(dòng)態(tài)綁定的,子類(lèi)LoggingWidget 的對(duì)象調(diào)用doSomething方法,一定是綁定到子類(lèi)自身的doSomething方法,必須用super關(guān)鍵字告訴虛擬機(jī),這里要調(diào)用的是父類(lèi)的doSomething方法。
實(shí)際上,如果我們分析運(yùn)行時(shí)的LoggingWidget 類(lèi),那我們看到的應(yīng)該是這樣子的(這里只是為了分析,真實(shí)情況肯定和下面的例子不同):
1 public class LoggingWidget extendsWidget {2 public synchronized voidWidget.doSomething() {3 //do something
4 } //父類(lèi)的doSomething方法
5
6 public synchronized voiddoSomething() {7 super.doSomething();8 }9 }
子類(lèi)對(duì)象,其實(shí)是持有父類(lèi)Widget 的doSomething方法的,只需要使用super關(guān)鍵字告訴虛擬機(jī)要運(yùn)行的是父類(lèi)的doSomething方法,虛擬機(jī)會(huì)去調(diào)用子類(lèi)對(duì)象中的父類(lèi)Widget 的doSomething方法的。所以,super關(guān)鍵字并沒(méi)有新建一個(gè)父類(lèi)的對(duì)象,比如說(shuō)widget,然后再去調(diào)用widget.doSomething方法,實(shí)際上調(diào)用父類(lèi)doSomething方法的還是我們的子類(lèi)對(duì)象。
那么這樣就很好理解了,如果一個(gè)線程有子類(lèi)對(duì)象的引用loggingWidget,然后調(diào)用loggingWidget.doSomething方法的時(shí)候,會(huì)請(qǐng)求子類(lèi)對(duì)象loggingWidget 的對(duì)象鎖;又因?yàn)閘oggingWidget 的doSomething方法中調(diào)用的父類(lèi)的doSomething方法,實(shí)際上還是要請(qǐng)求子類(lèi)對(duì)象loggingWidget 的對(duì)象鎖,那么如果synchronized關(guān)鍵字不是個(gè)可重入鎖的話,就會(huì)在子類(lèi)對(duì)象持有的父類(lèi)doSomething方法上產(chǎn)生死鎖了。正因?yàn)閟ynchronized關(guān)鍵字的可重入鎖,當(dāng)前線程因?yàn)橐呀?jīng)持有了子類(lèi)對(duì)象loggingWidget 的對(duì)象鎖,后面再遇到請(qǐng)求loggingWidget 的對(duì)象鎖就可以暢通無(wú)阻地執(zhí)行同步方法了。
更進(jìn)一步,將上面的示例代碼改寫(xiě)一下,那么就算synchronized不是可重入鎖,也不會(huì)產(chǎn)生死鎖的問(wèn)題了。代碼如下:
1 public classWidget {2 public synchronized voiddoSomething(){3 //do something
4 }5 }6 public class LoggingWidget extendsWidget {7 public synchronized voiddoSomething() {8 Widget widget = newWidget();9 widget.doSomething();10 }11 }
在子類(lèi)的doSomething方法中,直接新建了一個(gè)父類(lèi)的對(duì)象widget,然后用這個(gè)父類(lèi)對(duì)象來(lái)調(diào)用父類(lèi)的doSomething方法,實(shí)際上請(qǐng)求的是這個(gè)父類(lèi)對(duì)象widget的對(duì)象鎖,就不涉及到可重入鎖的問(wèn)題了。
總結(jié)
以上是生活随笔為你收集整理的java代码如何避免死锁,Java可重入锁如何避免死锁的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: php读取文本写入数据库,php读取tx
- 下一篇: java中exec命令,使用Java中的