Java多線程之死鎖詳解
1、死鎖
出現(xiàn)場景:當(dāng)線程A擁有了A對象的鎖,想要去獲取B對象的鎖;線程B擁有了B對象的鎖,想要擁有A對象的鎖,兩個線程在獲取鎖的時候,都不會釋放已經(jīng)持有的鎖,于是,就造成了死鎖。
示例代碼:
@Slf4j
public class ThreadTest {
private static Object objectA = new Object();
private static Object objectB = new Object();
public static void main(String[] args) throws InterruptedException {
Thread t2 = new Thread(()->{
synchronized (objectA){
log.debug("線程t2獲取到了objectA");
synchronized (objectB){
log.debug("線程t2獲取到了objectB");
}
}
},"t2");
Thread t1 = new Thread(()->{
synchronized (objectB){
log.debug("線程t1獲取到了objectB");
synchronized (objectA){
log.debug("線程t1獲取到了objectA");
}
}
},"t1");
t2.start();
t1.start();
}
}
如何檢測死鎖:
兩種方法
(1)找到本機jconsole程序,直接在windows系統(tǒng)搜索就可以,打開是這個樣子。

然后在本地進(jìn)程里面選擇你的進(jìn)程,其實就是你的項目名稱。然后點擊連接,在點擊不安全連接。

進(jìn)去之后點擊線程

再點擊檢測死鎖

最后就能看到死鎖的線程了

(2)首先是在idea的控制臺,打開Terminal,輸入【jps】命令查看所有的進(jìn)程id,找到你自己的java類名稱對應(yīng)的id。
然后輸入【jstack + 進(jìn)程號】 就可以查詢到該進(jìn)程的所有線程信息。在輸出信息的最下面,就可以看到如下圖所示的線程死鎖信息。

2、死鎖經(jīng)典問題——哲學(xué)家就餐問題
經(jīng)典場景:有四位哲學(xué)及在一正方形的桌子上面吃飯,桌子的每個角有一根筷子,一共四根,那么,當(dāng)每個哲學(xué)家都拿起自己左邊的筷子之后,再去拿自己右邊的筷子的時候,就會發(fā)現(xiàn)自己右邊沒有筷子,這時哲學(xué)就就會等右邊的哲學(xué)家放下筷子,但是每個哲學(xué)家都是這個想法,那么都不會放下筷子,并且都拿不到右邊的筷子,因此就造成了死鎖。
代碼實現(xiàn)例子:
@Slf4j
public class Thread1 {
public static void main(String[] args) throws InterruptedException {
//筷子對象
Chopsticks c1 = new Chopsticks("c1");
Chopsticks c2 = new Chopsticks("c2");
Chopsticks c3 = new Chopsticks("c3");
Chopsticks c4 = new Chopsticks("c4");
new Philosopher("李云龍",c1,c2).start();
new Philosopher("趙剛",c2,c3).start();
new Philosopher("魏和尚",c3,c4).start();
new Philosopher("張大彪",c4,c1).start();
}
}
//筷子
class Chopsticks{
private String name;
public Chopsticks(String name) {
this.name = name;
}
}
//哲學(xué)家
@Slf4j
class Philosopher extends Thread{
//名字
private String name;
//筷子
private Chopsticks left;
private Chopsticks right;
public Philosopher(String name, Chopsticks left, Chopsticks right) {
super(name);
this.left = left;
this.right = right;
}
@Override
public void run() {
while(true){
synchronized (right){
synchronized (left){
eat(name);
}
}
}
}
private void eat(String name){
log.debug(name + "正在吃飯");
}
}
測試結(jié)果:可以實現(xiàn)吃飯操作,但是會出現(xiàn)場景中描述的問題,出現(xiàn)線程死鎖。

總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
Java使用正則表達(dá)式判斷獨立字符的存在(代碼示例)
通過使用正則表達(dá)式,我們可以更加靈活地判斷字符串中是否包含特定的字符,并且可以控制匹配的條件,如獨立的字符,這為我們處理字符串提供了更多的選擇和功能,這篇文章主要介紹了Java使用正則表達(dá)式判斷獨立字符的存在,需要的朋友可以參考下2023-10-10
SpringBoot整合Quartz實現(xiàn)定時任務(wù)詳解
這篇文章主要介紹了Java?任務(wù)調(diào)度框架?Quartz,Quartz是OpenSymphony開源組織在Job?scheduling領(lǐng)域又一個開源項目,完全由Java開發(fā),可以用來執(zhí)行定時任務(wù),類似于java.util.Timer。,下面我們來學(xué)習(xí)一下關(guān)于?Quartz更多的詳細(xì)內(nèi)容,需要的朋友可以參考一下2022-08-08
Spring Boot數(shù)據(jù)庫鏈接池配置方法
這篇文章主要介紹了Spring Boot數(shù)據(jù)庫鏈接池配置方法,需要的朋友可以參考下2017-04-04
利用HttpUrlConnection 上傳 接收文件的實現(xiàn)方法
下面小編就為大家?guī)硪黄肏ttpUrlConnection 上傳 接收文件的實現(xiàn)方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-11-11

