【併發編程】死鎖

原文:慕課網實戰·高併發探索(十五):線程死鎖html

什麼是死鎖?

通俗的說,死鎖就是兩個或者多個線程,相互佔用對方須要的資源,而都不進行釋放,致使彼此之間都相互等待對方釋放資源,產生了無限制等待的現象。死鎖一旦發生,若是沒有外力介入,這種等待將永遠存在,從而對程序產生嚴重影響。java

用來描述死鎖的問題最有名的場景就是「哲學家就餐問題」。哲學家就餐問題能夠這樣表述:假設有五位哲學家圍坐在一張圓形餐桌旁,作如下兩件事之一:吃飯或者思考。吃東西的時候他們就中止思考,思考的時候也中止吃東西。餐桌中間有一大碗意大利麪,每兩個哲學家之間有一隻餐叉。由於只用一隻餐叉很難吃到意大利麪,因此假設哲學家必須用兩隻餐叉吃東西。他們只能使用本身左右手邊的那兩隻餐。哲學家歷來不交談,這就跟危險,可能產生死鎖,每一個哲學家都拿着左手的餐叉永遠等右邊的餐叉(或者相反)….編程

死鎖產生的4個必要條件

互斥條件:進程對鎖分配的資源進行排他性使用
請求和保持條件:線程已經保持了一個資源,可是又提出了其餘請求,而該資源已被其餘線程佔用
不剝奪條件:在使用時不能被剝奪,只能本身用完釋放
環路等待條件:資源調用是一個環形的鏈windows

死鎖示例併發

@Slf4j
public class DeadLock implements Runnable {
    public int flag = 1;
    //靜態對象是類的全部對象共享的
    private static Object o1 = new Object(), o2 = new Object();

    @Override
    public void run() {
        log.info("flag:{}", flag);
        if (flag == 1) {
            synchronized (o1) {
                try {
                    Thread.sleep(500);
                } catch (Exception e) {
                    e.printStackTrace();
                }
                synchronized (o2) {
                    log.info("1");
                }
            }
        }
        if (flag == 0) {
            synchronized (o2) {
                try {
                    Thread.sleep(500);
                } catch (Exception e) {
                    e.printStackTrace();
                }
                synchronized (o1) {
                    log.info("0");
                }
            }
        }
    }

    public static void main(String[] args) {
        DeadLock td1 = new DeadLock();
        DeadLock td2 = new DeadLock();
        td1.flag = 1;
        td2.flag = 0;
        //td1,td2都處於可執行狀態,但JVM線程調度先執行哪一個線程是不肯定的。
        //td2的run()可能在td1的run()以前運行
        new Thread(td1).start();
        new Thread(td2).start();
    }
}

上述代碼出現死鎖緣由:ide

當DeadLock類的對象flag==1時(td1),先鎖定o1,睡眠500毫秒
而td1在睡眠的時候另外一個flag==0的對象(td2)線程啓動,先鎖定o2,睡眠500毫秒
td1睡眠結束後須要鎖定o2才能繼續執行,而此時o2已被td2鎖定;
td2睡眠結束後須要鎖定o1才能繼續執行,而此時o1已被td1鎖定;
td一、td2相互等待,都須要獲得對方鎖定的資源才能繼續執行,從而死鎖。高併發

確認死鎖

在真實的環境中,咱們發現程序沒法執行,而且CPU佔用爲0,這樣就有理由懷疑產生了死鎖,可是光懷疑是不行的,咱們須要一個實際的驗證方法。接下來咱們使用jdk提供的工具來檢測是否真正發生了死鎖。
運行上述的代碼,並在windows系統中使用cmd進入控制檯,輸入如下命令:工具

jps

可見控制檯輸出:咱們上邊運行的類的類名以及對應的進程ID
.net

接下來使用命令獲取進程對應線程的堆棧信息:線程

jstack 9284

分析堆棧信息(提取有用的部分)

兩個線程都進行了加鎖操做(如上圖)

系統發現了一個Java-level的線程死鎖。確認無疑是發生了死鎖現象。

避免死鎖

相關文章
相關標籤/搜索