java避免死鎖的常見(jiàn)方法代碼解析
死鎖
索是一個(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ì)遇到這樣的問(wèn)題,比如 thread_1 拿到索之后,因?yàn)橐恍┊惓G闆r沒(méi)有釋放索(死循環(huán))。又或者是 thread_1 拿到一個(gè)數(shù)據(jù)庫(kù)索,釋放鎖的時(shí)候拋出了異常,沒(méi)釋放掉。
一旦出現(xiàn)死鎖,業(yè)務(wù)是可感知的,因?yàn)椴荒芾^續(xù)提供服務(wù)了,那么只能通過(guò)dump 線程查看到底是哪個(gè)線程出現(xiàn)了問(wè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è)常見(jiàn)方法。
避免一個(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://www.dbjr.com.cn/article/131946.htm
http://www.dbjr.com.cn/article/131943.htm
總結(jié)
以上就是本文關(guān)于java避免死鎖的常見(jiàn)方法代碼解析的全部?jī)?nèi)容,希望對(duì)大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站其他相關(guān)專題,如有不足之處,歡迎留言指出。感謝朋友們對(duì)本站的支持!
相關(guān)文章
MybatisPlus開(kāi)啟二級(jí)緩存的方法詳解
這篇文章主要介紹了MybatisPlus開(kāi)啟二級(jí)緩存的方法詳解,二級(jí)緩存是基于mapper文件的namespace級(jí)別,也就是說(shuō)多個(gè)sqlSession可以共享一個(gè)mapper中的二級(jí)緩存區(qū)域,需要的朋友可以參考下2023-11-11
idea?intellij快速修復(fù)if語(yǔ)句缺少大括號(hào)的問(wèn)題
這篇文章主要介紹了idea?intellij快速修復(fù)if語(yǔ)句缺少大括號(hào)的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-04-04
JavaWeb如何實(shí)現(xiàn)統(tǒng)一查詢接口(jfinal)
這篇文章主要介紹了JavaWeb如何實(shí)現(xiàn)統(tǒng)一查詢接口(jfinal),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-06-06
Java中如何動(dòng)態(tài)創(chuàng)建接口的實(shí)現(xiàn)方法
這篇文章主要介紹了Java中如何動(dòng)態(tài)創(chuàng)建接口的實(shí)現(xiàn)方法的相關(guān)資料,需要的朋友可以參考下2017-09-09
Java中try-catch的使用及注意細(xì)節(jié)
現(xiàn)在有很多的語(yǔ)言都支持try-catch,比如常見(jiàn)的就是c++,java等,這篇文章主要給大家介紹了關(guān)于Java中try-catch的使用及注意細(xì)節(jié)的相關(guān)資料,文中通過(guò)圖文以及實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-06-06
關(guān)于springboot配置druid數(shù)據(jù)源不生效問(wèn)題(踩坑記)
今天日常跟著網(wǎng)課學(xué)習(xí),學(xué)到了整合druid數(shù)據(jù)源,遇到了好幾個(gè)坑,希望這篇文章可以幫助一些和我一樣踩坑的人2021-09-09
關(guān)于ApplicationContext的三個(gè)常用實(shí)現(xiàn)類
這篇文章主要介紹了關(guān)于ApplicationContext的三個(gè)常用實(shí)現(xiàn)類,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-06-06

