欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java多線程中的死鎖詳解

 更新時間:2023年08月26日 08:29:48   作者:一只愛打拳的程序猿  
這篇文章主要介紹了Java多線程中的死鎖詳解,死鎖是指兩個或多個線程在執(zhí)行過程中,因爭奪資源而造成的一種互相等待的現(xiàn)象,如果不提前預防或外界干擾,這些線程將無法執(zhí)行下去,需要的朋友可以參考下

前言

死鎖(Deadlock)是指兩個或多個線程在執(zhí)行過程中,因爭奪資源而造成的一種互相等待的現(xiàn)象

如果不提前預防或外界干擾,這些線程將無法執(zhí)行下去。

因此,本篇博文講解造成死鎖的原因以及解決方案。

1. 造成死鎖的原因

死鎖是多個線程在相互等待對方所占有的鎖才能繼續(xù)執(zhí)行下去,造成線程都被阻塞都無法執(zhí)行下去,也釋放不了線程自己的鎖資源,造成僵局。

舉例說明:

我和朋友去拉面館吃面,我先拿起了辣椒他拿起了醋。此時他想要辣椒、我想要醋,但我倆誰也不讓誰。

我對他說:你先把辣椒給我,我用完了就把醋給你。他對我說:你先把醋給我,我用完了就把辣椒給你。

我和他也不讓誰,最后導致死鎖。在上述例子中,辣椒和醋就是兩個鎖,我和他就是兩個線程。關于死鎖會引申的一個哲學家就餐問題。

2. 哲學家就餐問題

哲學家就餐問題是一個經(jīng)典的并發(fā)編程問題,它描述的是五個哲學家圍坐在一張圓形餐桌旁,每個哲學家面前有一碗飯和一只筷子,碗和筷子都是共享資源。

哲學家需要用一雙筷子來吃飯,但是只有相鄰的兩個哲學家之間有一只筷子,因此他們必須在餐桌上競爭筷子,才能拿到一雙筷子吃飯。

這個問題會引發(fā)死鎖(Deadlock)和饑餓(Starvation)等并發(fā)編程中的經(jīng)典問題。

如果所有哲學家同時都想獲取餐具,但每個哲學家只能等待旁邊的哲學家放下筷子才能獲取餐具,這可能會導致死鎖。

如果所有哲學家只有在其他人都吃到飯才能拿到自己的餐具,那么有些哲學家可能會因為等待時間太長而饑餓。

代碼案例:

有兩個哲學家在進行就餐,哲學家1獲取到筷子1后繼續(xù)嘗試獲取筷子2,哲學家2獲取到筷子2后繼續(xù)嘗試獲取筷子1,造成了死鎖狀態(tài)。

public class DeadlockDemo {
        public static void main(String[] args) {
            Object obj1 = new Object();//筷子1
            Object obj2 = new Object();//筷子2
            // 線程1獲取obj1,然后等待obj2
            Thread thread1 = new Thread(() -> {
                synchronized (obj1) {
                    System.out.println(Thread.currentThread().getName() + " 獲取 obj1鎖");
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    synchronized (obj2) {
                        System.out.println(Thread.currentThread().getName() + " 獲取 obj2鎖");
                    }
                }
            }, "哲學家 1");
            // 線程2獲取obj2,然后等待obj1
            Thread thread2 = new Thread(() -> {
                synchronized (obj2) {
                    System.out.println(Thread.currentThread().getName() + " 獲取 obj2鎖");
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    synchronized (obj1) {
                        System.out.println(Thread.currentThread().getName() + " 獲取 obj1鎖");
                    }
                }
            }, "哲學家 2");
            t1.start();
            t2.start();
        }
}

運行后打?。?/p>

上述代碼,中 thread1 是哲學家1,thread2 是哲學家2,obj1 鎖和 obj2 鎖分別代表兩只筷子。

哲學家1 占用了 obj1 鎖,哲學家2 占用了 obj2 鎖。

此時 哲學家1 在擁有 obj1 時獲取 obj2 是獲取不到的,哲學家2 在擁有 obj2 時獲取 obj1 也是獲取不到的。

這樣就會造成死鎖。也就是本篇文章開頭所說的,一個線程處于永久獲取到鎖的狀態(tài),其他線程無限阻塞等待,這就造成了死鎖。

2.1 造成死鎖的原因

  • 互斥:一個資源只能被一個進程獨占,即該資源只能被一個進程占有或使用。
  • 請求和保持:進程已經(jīng)保持了至少一個資源,并還在嘗試獲取其他進程保持的資源。
  • 不可搶占:某個資源只能由占有它的線程釋放,不能被其他線程強制占有或搶占。
  • 循環(huán)等待:假設一個隊列有{t1,t2,t3}這三個線程,t1 等待 t2 所占資源,t2 等待 t3 所占資源,t3 等待 t1 所占資源,這樣就會造成循環(huán)等待。 

2.2 解決死鎖問題

解決死鎖問題,最好的方式就是預防死鎖。我們把鎖按照順序進行編號,并且約定每個線程在獲取多把鎖的時候必須先獲取編號小的,后獲取編號大的。這樣就不會形成死鎖了。

因為,最先獲取編號小鎖的線程會優(yōu)先完成相應任務。輪到其他線程再獲取小編號鎖時,鎖就處于空閑狀態(tài)了。

代碼案例:

public class DeadlockDemo {
        public static void main(String[] args) {
            Object obj1 = new Object();//筷子1
            Object obj2 = new Object();//筷子2
            Thread t1 = new Thread(() -> {
                synchronized (obj1) {
                    System.out.println(Thread.currentThread().getName() + " 獲取 obj1鎖");
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    synchronized (obj2) {
                        System.out.println(Thread.currentThread().getName() + " 獲取 obj2鎖");
                    }
                }
            }, "哲學家 1");
            Thread t2 = new Thread(() -> {
                synchronized (obj1) {
                    System.out.println(Thread.currentThread().getName() + " 獲取 obj2鎖");
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    synchronized (obj2) {
                        System.out.println(Thread.currentThread().getName() + " 獲取 obj1鎖");
                    }
                }
            }, "哲學家 2");
            t1.start();
            t2.start();
        }
}

運行后打?。?/p>

在上述代碼中,我們創(chuàng)建了兩個鎖對象 obj1 和 obj2 分別代表兩只筷子,并創(chuàng)建兩個線程 thread1 和 thread 2 分別代表兩個哲學家。

當 哲學家1 首先獲取到 obj1 后,由于 obj2 并未線程占用,則立馬獲取 obj2 ,獲得了兩雙筷子,就餐后釋放 obj1 和 obj2。

這時 哲學家2 嘗試獲取 obj2 和 obj1 時,發(fā)現(xiàn)兩雙筷子都沒人使用,直接開始就餐。

總結(jié)

什么是死鎖?

當多個線程并發(fā)后,線程在占有鎖資源的情況下還嘗試獲取別的線程所占有的鎖資源。

這樣就會操作線程永遠獲取不到其他鎖資源,也不能釋放本身鎖資源。

從而形成的一種僵局,這就是死鎖。

造成死鎖的原因是什么?

造成死鎖的原因有,互斥、請求和保持、不可搶占、循環(huán)等待。(文章中有講解)

死鎖的解決方案

我們可以對多個鎖按照順序進行編號,并約定每個線程在使用多把鎖時優(yōu)先使用最小編號的鎖,這樣就能保證其他線程再獲取小編號鎖時,鎖資源已得到釋放。

到此這篇關于Java多線程中的死鎖詳解的文章就介紹到這了,更多相關Java多線程死鎖內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

最新評論