java避免死鎖的常見方法代碼解析
死鎖
索是一個(gè)非常有用的工具,運(yùn)用場(chǎng)景非常多,因?yàn)樗褂闷饋?lái)非常簡(jiǎn)單,而且易于理解。但同時(shí)它也會(huì)帶來(lái)一些困擾,那就是可能會(huì)引起死鎖,一旦產(chǎn)生死鎖,就會(huì)造成系統(tǒng)功能不可用。讓我們先來(lái)看一段代碼,這段代碼會(huì)引起死鎖,使線程 thread_1 和線程 thread_2 互相等待對(duì)方釋放鎖。
package thread; public class DeadLockDemo { private static String A = "A"; private static String B = "B"; public static void main(String args[]) { new DeadLockDemo().deadLock(); } private void deadLock() { // 線程thread_1 Thread thread_1 = new Thread(new Runnable() { @Override public void run() { synchronized (A) { System.err.println("--thread_1 lock A----"); synchronized (B) { System.err.println("--thread_1 lock B----"); } } } } ); // 線程thread_2 Thread thread_2 = new Thread(new Runnable() { @Override public void run() { synchronized (B) { System.out.println("--thread_2 lock B----"); synchronized (A) { System.out.println("--thread_2 lock A----"); } } } } ); thread_1.start(); thread_2.start(); } }
這段代碼只是演示死鎖的場(chǎng)景,在現(xiàn)實(shí)中你可能不會(huì)寫出這樣的代碼。但是在一些更為復(fù)雜的場(chǎng)景中,你可能會(huì)遇到這樣的問題,比如 thread_1 拿到索之后,因?yàn)橐恍┊惓G闆r沒有釋放索(死循環(huán))。又或者是 thread_1 拿到一個(gè)數(shù)據(jù)庫(kù)索,釋放鎖的時(shí)候拋出了異常,沒釋放掉。
一旦出現(xiàn)死鎖,業(yè)務(wù)是可感知的,因?yàn)椴荒芾^續(xù)提供服務(wù)了,那么只能通過(guò)dump 線程查看到底是哪個(gè)線程出現(xiàn)了問題,以下線程信息告訴我們是 DeadLockDemo類的第 35 行和21行引起了死鎖。
"Thread-1" prio=6 tid=0x000000000cb13800 nid=0x19ac waiting for monitor entry [0 x000000000d67f000] java.lang.Thread.State: BLOCKED (on object monitor) at thread.DeadLockDemo$2.run(DeadLockDemo.java:35) - waiting to lock <0x00000007d5a9be88> (a java.lang.String) - locked <0x00000007d5a9beb8> (a java.lang.String) at java.lang.Thread.run(Unknown Source) "Thread-0" prio=6 tid=0x000000000cb0e800 nid=0x6bc waiting for monitor entry [0x 000000000d48f000] java.lang.Thread.State: BLOCKED (on object monitor) at thread.DeadLockDemo$1.run(DeadLockDemo.java:21) - waiting to lock <0x00000007d5a9beb8> (a java.lang.String) - locked <0x00000007d5a9be88> (a java.lang.String) at java.lang.Thread.run(Unknown Source)
避免死鎖的幾個(gè)常見方法。
避免一個(gè)線程同時(shí)獲取多個(gè)鎖。
避免一個(gè)線程在索內(nèi)同時(shí)占用多個(gè)資源,盡量保證每個(gè)索只占用一個(gè)資源。
嘗試使用定時(shí)索,使用 lock.tryLock(timeout) 來(lái)替代使用內(nèi)部索機(jī)制。
對(duì)于數(shù)據(jù)庫(kù)索,加鎖和解鎖必須在一個(gè)數(shù)據(jù)庫(kù)連接里,否則會(huì)出現(xiàn)解鎖失敗的情況。
referance:
http://chabaoo.cn/article/131946.htm
http://chabaoo.cn/article/131943.htm
總結(jié)
以上就是本文關(guān)于java避免死鎖的常見方法代碼解析的全部?jī)?nèi)容,希望對(duì)大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站其他相關(guān)專題,如有不足之處,歡迎留言指出。感謝朋友們對(duì)本站的支持!
相關(guān)文章
Java的this關(guān)鍵字的使用與方法的重載相關(guān)知識(shí)
這篇文章主要介紹了Java的this關(guān)鍵字的使用與方法的重載相關(guān)知識(shí),是Java入門學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下2015-09-09SpringBoot打包成Docker鏡像的幾種實(shí)現(xiàn)方式
Spring Boot是一個(gè)用于構(gòu)建獨(dú)立的、可執(zhí)行的Spring應(yīng)用程序的框架,結(jié)合使用Spring Boot和Docker,可以方便地將應(yīng)用程序部署到不同的環(huán)境中本文,主要介紹了SpringBoot打包成Docker鏡像的幾種實(shí)現(xiàn)方式,感興趣的可以了解一下2024-01-01java簡(jiǎn)單實(shí)現(xiàn)八叉樹圖像處理代碼示例
這篇文章主要介紹了java簡(jiǎn)單實(shí)現(xiàn)八叉樹圖像處理代碼示例,具有一定借鑒價(jià)值,需要的朋友可以參考下。2017-12-12使用spring stream發(fā)送消息代碼實(shí)例
這篇文章主要介紹了使用spring stream發(fā)送消息代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-05-05SpringMVC域?qū)ο蠊蚕頂?shù)據(jù)示例詳解
這篇文章主要為大家介紹了SpringMVC域?qū)ο蠊蚕頂?shù)據(jù)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05解決MyBatis-Plus使用動(dòng)態(tài)表名selectPage不生效的問題
這篇文章主要介紹了如惡化解決MyBatis-Plus使用動(dòng)態(tài)表名selectPage不生效的問題,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-11-11Java 中的 DataInputStream 介紹_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
DataInputStream 是數(shù)據(jù)輸入流。它繼承于FilterInputStream。接下來(lái)通過(guò)本文給大家介紹Java 中的 DataInputStream的相關(guān)知識(shí),需要的朋友參考下吧2017-05-05基于SpringBoot實(shí)現(xiàn)QQ郵箱驗(yàn)證碼注冊(cè)功能
QQ 郵箱是由騰訊公司推出的一款免費(fèi)郵箱服務(wù),它提供了完整的郵件發(fā)送和接收功能,并且還支持多種郵件格式和附件類型,QQ 郵箱還具有強(qiáng)大的反垃圾郵件功能,可以有效地過(guò)濾垃圾郵件,并保護(hù)用戶隱私和安全,所以本文給大家介紹了基于SpringBoot實(shí)現(xiàn)QQ郵箱驗(yàn)證碼注冊(cè)功能2024-11-11