Java線程死鎖實(shí)例及解決方法
這篇文章主要介紹了Java線程死鎖實(shí)例及解決方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
1、死鎖的定義
所謂死鎖是指多個(gè)線程因競(jìng)爭(zhēng)資源而造成的一種僵局(互相等待),若無(wú)外力作用,這些進(jìn)程都將無(wú)法向前推進(jìn)
2、死鎖產(chǎn)生的必要條件
互斥條件:線程要求對(duì)所分配的資源(如打印機(jī))進(jìn)行排他性控制,即在一段時(shí)間內(nèi)某資源僅為一個(gè)線程所占有。此時(shí)若有線程請(qǐng)求該資源,則請(qǐng)求線程只能等待。
不剝奪條件:線程所獲得的資源在未使用完畢之前,不能被其他線程傾向奪走,即只能由獲得該資源的線程自己來(lái)釋放(只能是主動(dòng)釋放)。
請(qǐng)求和保持條件:線程已經(jīng)保持了至少一個(gè)資源,但又提出了新的資源請(qǐng)求,而該線程已被其他線程占有,此時(shí)請(qǐng)求進(jìn)程被阻塞,但對(duì)自己已獲得的資源保持不放。
循環(huán)等待條件:存在一種線程資源的循環(huán)等待鏈,鏈中每一個(gè)線程已獲得的資源同時(shí)被鏈中下一個(gè)線程所請(qǐng)求。即存在一個(gè)處于等待狀態(tài)的線程集合{P1,P2,...,Pn},其中Pi等待的資源被P(i+1)占有(i=0,1,..,n-1),Pn等待的資源被P0占有,如下圖。

3、產(chǎn)生死鎖的一個(gè)例子
/**
* 一個(gè)簡(jiǎn)單的死鎖類(lèi)
* 當(dāng)DeadLock類(lèi)的對(duì)象flag==1時(shí)(td1),先鎖定o1,睡眠500ms
* 而td1在睡眠的時(shí)候另一個(gè)flag==0的對(duì)象(td2)線程啟動(dòng),先鎖定o2,睡眠500ms
* td1 睡眠結(jié)束后需要鎖定 o2 才能繼續(xù)執(zhí)行,而此時(shí) o2 已被 td2 鎖定;
* td2 睡眠結(jié)束后需要鎖定 o1 才能繼續(xù)執(zhí)行,而此時(shí) o1 已被 td1 鎖定;
* td1、td2 相互等待,都需要得到對(duì)方鎖定的資源才能繼續(xù)執(zhí)行,從而死鎖。
*/
public class DeadLock implements Runnable {
public int flag = 1;
//靜態(tài)對(duì)象是類(lèi)的所有對(duì)象共享的
private static Object o1 = new Object(), o2 = new Object();
@Override
public void run() {
System.out.println("flag="+flag);
if(flag==1){
synchronized (o1){
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
synchronized (o2){
System.out.println("1");
}
}
}
if(flag==0){
synchronized (o2){
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
synchronized (o1){
System.out.println("0");
}
}
}
}
public static void main(String[] args) {
DeadLock td1 = new DeadLock();
DeadLock td2 = new DeadLock();
td1.flag=1;
td2.flag=0;
//td1,td2都處于可執(zhí)行狀態(tài),但JVM線程調(diào)度先執(zhí)行哪個(gè)線程是不確定的。
//td2的run()可能在td1的run()之前運(yùn)行
new Thread(td1).start();
new Thread(td2).start();
}
}

4、如何避免死鎖
在有些情況下死鎖是可以避免的。兩種用于避免死鎖的技術(shù)
1)加鎖順序(線程按照一定的順序加鎖

2)加鎖時(shí)限(線程獲取鎖的時(shí)候加上一定的時(shí)限,超過(guò)時(shí)限則放棄對(duì)該鎖的請(qǐng)求,并釋放自己占有的鎖)

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java配置JDK開(kāi)發(fā)環(huán)境及環(huán)境變量
這篇文章主要為大家詳細(xì)介紹了Java配置JDK開(kāi)發(fā)環(huán)境及環(huán)境變量,文中安裝步驟介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-09-09
java實(shí)現(xiàn)KFC點(diǎn)餐系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)KFC點(diǎn)餐系統(tǒng),模擬肯德基快餐店的收銀系統(tǒng),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01
java實(shí)現(xiàn)根據(jù)pdf模板生成文件并插入文字,選項(xiàng),簽名和公章
這篇文章主要為大家詳細(xì)介紹了java如何實(shí)現(xiàn)根據(jù)pdf模板生成文件并插入文字,選項(xiàng),簽名和公章,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-11-11
一文詳解Java應(yīng)用頻繁Full GC的原因與調(diào)優(yōu)方案
在高并發(fā)的交易場(chǎng)景中,Java應(yīng)用的性能穩(wěn)定性直接影響用戶(hù)體驗(yàn),近期某交易平臺(tái)訂單服務(wù)在高峰期頻繁出現(xiàn)Full GC,導(dǎo)致服務(wù)暫停數(shù)秒,給用戶(hù)帶來(lái)了極差的使用感受,本文將結(jié)合該服務(wù)的實(shí)際場(chǎng)景,詳細(xì)講解Full GC的常見(jiàn)原因與調(diào)優(yōu)方案,需要的朋友可以參考下2025-09-09
springboot對(duì)數(shù)據(jù)庫(kù)密碼加密的實(shí)現(xiàn)
這篇文章主要介紹了springboot對(duì)數(shù)據(jù)庫(kù)密碼加密的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12
Spring的異常處理@ExceptionHandler注解解析
這篇文章主要介紹了Spring的異常處理@ExceptionHandler注解解析,當(dāng)一個(gè)Controller中有方法加了@ExceptionHandler之后,這個(gè)Controller其他方法中沒(méi)有捕獲的異常就會(huì)以參數(shù)的形式傳入加了@ExceptionHandler注解的那個(gè)方法中,需要的朋友可以參考下2023-12-12
MyBatisPlus+Lombok實(shí)現(xiàn)分頁(yè)功能的方法詳解
Lombok是一個(gè)Java類(lèi)庫(kù),提供了一組注解,簡(jiǎn)化POJO實(shí)體類(lèi)開(kāi)發(fā)。本文將為大家介紹一下Lombok的使用以及如何利用MyBatisPlus+Lombok實(shí)現(xiàn)分頁(yè)功能,感興趣的可以動(dòng)手嘗試一下2022-07-07

