溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java線程技術(shù)中的死鎖問(wèn)題怎么解決

發(fā)布時(shí)間:2022-01-11 10:12:59 來(lái)源:億速云 閱讀:141 作者:iii 欄目:編程語(yǔ)言

這篇文章主要介紹“Java線程技術(shù)中的死鎖問(wèn)題怎么解決”的相關(guān)知識(shí),小編通過(guò)實(shí)際案例向大家展示操作過(guò)程,操作方法簡(jiǎn)單快捷,實(shí)用性強(qiáng),希望這篇“Java線程技術(shù)中的死鎖問(wèn)題怎么解決”文章能幫助大家解決問(wèn)題。

我們知道,使用 synchronized 關(guān)鍵字可以有效的解決線程同步問(wèn)題,但是如果不恰當(dāng)?shù)氖褂?synchronized 關(guān)鍵字的話也會(huì)出問(wèn)題,即我們所說(shuō)的死鎖。死鎖是這樣一種情形:多個(gè)線程同時(shí)被阻塞,它們中的一個(gè)或者全部都在等待某個(gè)資源被釋放。由于線程被無(wú)限期地阻塞,因此程序不可能正常終止。

我們先寫一個(gè)死鎖的例子,再來(lái)分析一下死鎖產(chǎn)生的原因:

public class DeadLock {
    public static void main(String[] args) {
        Business business = new Business1();
        //開啟一個(gè)線程執(zhí)行Business類中的functionA方法
        new Thread(new Runnable() {         
            @Override
            public void run() {
                while(true) {
                    business.functionA();
                }
            }
        }).start();
        //開啟另一個(gè)線程執(zhí)行Business類中的functionB方法
        new Thread(new Runnable() {         
            @Override
            public void run() {
                while(true) {
                    business.functionB();
                }
            }
        }).start();
    }
}
class Business { //定義兩個(gè)鎖,兩個(gè)方法
    //定義兩個(gè)鎖
    public static final Object lock_a = new Object();
    public static final Object lock_b = new Object();   
    public void functionA() {
        synchronized(lock_a) {
            System.out.println("---ThreadA---lock_a---");
            synchronized(lock_b) {
                System.out.println("---ThreadA---lock_b---");
            }
        }
    }
    public void functionB() {
        synchronized(lock_b) {
            System.out.println("---ThreadB---lock_b---");
            synchronized(lock_a) {
                System.out.println("---ThreadB---lock_a---");
            }
        }
    }
}

程序的結(jié)構(gòu)很清晰,沒(méi)什么難度,先看一下程序的執(zhí)行結(jié)果:

---ThreadA---locka---
---ThreadA---lockb---
---ThreadA---locka---
---ThreadA---lockb---
---ThreadA---locka---
---ThreadA---lockb---
---ThreadA---locka---
---ThreadB---lockb---

從執(zhí)行結(jié)果來(lái)看,線程 A 跑著跑著,當(dāng)線程 B 一跑,啪嘰一下就掛了。我們來(lái)分析一下原因:從上面的代碼中可以看出,定義了一個(gè)類 Business,該類中維護(hù)了兩個(gè)鎖和兩個(gè)方法,每個(gè)方法都是 synchronized 連環(huán)套,并且使用的是不同的鎖。好了,現(xiàn)在 main 方法中開啟兩個(gè)線程 A 和 B,分別執(zhí)行 Business 類中的兩個(gè)方法。A 優(yōu)先執(zhí)行,跑的很爽,當(dāng) B 線程也開始執(zhí)行的時(shí)候,問(wèn)題來(lái)了,從執(zhí)行結(jié)果的最后兩行來(lái)看,A 線程進(jìn)入了 functionA方法中的第一個(gè) synchronized,拿到了 lock_a 鎖,B 線程進(jìn)入了 functionB 中的第一個(gè) synchronized,拿到了 lock_b 鎖,并且兩者的鎖都還沒(méi)釋放。接下來(lái)就是關(guān)鍵了:A 線程進(jìn)入第二個(gè) synchronized 的時(shí)候,發(fā)現(xiàn) lock_b 正在被 B 占用,那沒(méi)辦法,它只好被阻塞,等唄~同樣地,B 線程進(jìn)入第二個(gè) synchronized 的時(shí)候,發(fā)現(xiàn) lock_a 正在被 A 占用,那沒(méi)辦法,它也只好被阻塞,等唄~好了,兩個(gè)就這樣互相等著,你不放,我也不放……死了……

上面這個(gè)程序?qū)τ诶斫馑梨i很有幫助,因?yàn)榻Y(jié)構(gòu)很好,不過(guò)個(gè)人感覺(jué)這個(gè)死的還不過(guò)癮,因?yàn)閮蓚€(gè)線程是實(shí)現(xiàn)了兩個(gè)不同的 Runnable 接口,只不過(guò)調(diào)用了同一個(gè)類的兩個(gè)方法而已,因?yàn)槲野岩降姆椒ǚ诺揭粋€(gè)類中了。下面我把程序改一下,把要同步的代碼放到一個(gè) Runnable 中,讓它一運(yùn)行就掛掉……

public class DeadLock {    
    public static void main(String[] args) {            
        //開啟兩個(gè)線程,分別扔兩個(gè)自定義的Runnable進(jìn)去
        new Thread(new MyRunnable(true)).start();;
        new Thread(new MyRunnable(false)).start();;
    }
}
class MyRunnable implements Runnable
{
    private boolean flag; //用于判斷,執(zhí)行不同的同步代碼塊 
    MyRunnable(boolean flag) { //構(gòu)造方法
        this.flag = flag;
    }
    @Override
    public void run()
    {
        if(flag)
        {
            while(true){            
                synchronized(MyLock.lock_a)
                {
                    System.out.println("--threadA---lock_a--");
                    synchronized(MyLock.lock_b)
                    {
                        System.out.println("--threadA---lock_b--");
                    }   
                }
            }
        }
        else
        {
            while(true){            
                synchronized(MyLock.lock_b)
                {
                    System.out.println("--threadB---lock_a--");
                    synchronized(MyLock.lock_a)
                    {
                        System.out.println("--threadB---lock_b--");
                    }   
                }
            }
        }
    }
}
class MyLock //把兩把鎖放到一個(gè)類中定義,是為了兩個(gè)線程使用的都是這兩把鎖
{
    public static final Object lock_a = new Object();
    public static final Object lock_b = new Object();   
}

這個(gè)死鎖就厲害了,一運(yùn)行,啪嘰一下直接就掛掉了……看下運(yùn)行結(jié)果:

--threadA---locka--
--threadB---lockb--

關(guān)于“Java線程技術(shù)中的死鎖問(wèn)題怎么解決”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí),可以關(guān)注億速云行業(yè)資訊頻道,小編每天都會(huì)為大家更新不同的知識(shí)點(diǎn)。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI