Java多線程環境下死鎖模擬

1、死鎖產生的條件

  • 互斥:一次隻有一個進程可以使用一個資源。其他進程不能訪問已分配給其他進程的資源。
  • 不可搶占:不能搶占進程已占有的資源
  • 請求和保持:當一個進程等待其他進程釋放資源時,繼續占有已經分配的資源
  • 循環等待:存在一個封閉的進程鏈,使得每個進程至少占有此鏈中下一個進程所需要的一個資源。

註意:前三個條件都隻是死鎖存在的必要條件,但不是充分條件。第四個條件是充分條件。以上條件同樣適用於線程。

2、模擬多線程環境下死鎖的產生

/**
 * 死鎖產生的四個條件:互斥 請求和保持 不可搶占 循環等待
 * synchronized鎖住對象資源,保證互斥和不可搶占
 * 先持有一個資源,再去申請另一個資源->請求和保持
 */
public class DeadLock {

    // 模擬兩個資源
    public static Object lock1 = new Object();
    public static Object lock2 = new Object();

    // 先申請lock1資源,再去申請lock2資源
    public static void getLock1First(){
        synchronized (lock1){
            System.out.println("getlock1");
            getLock2();
        }
    }

    public static void getLock2(){
        synchronized (lock2) {
            System.out.println("lock1->lock2");
        }
    }

    // 先申請lock2資源,再申請lock1資源
    public static void getlock2First(){
        synchronized (lock2){
            System.out.println("getlock2");
            getLock1();
        }
    }

    public static void getLock1(){
        synchronized (lock1){
            System.out.println("lock2->lock1");
        }
    }

    public static void main(String[] args) {
        new Thread(){
            @Override
            public void run() {
                // 太難把控線程運行的時機瞭 也有可能此線程執行完瞭lock1->lock2的順序
                // 直接死循環一直運行
                // 總會剛好遇到線程1剛好獲得lock1,線程2剛好獲得lock2
                while (true){
                    getLock1First();
                }
            }
        }.start();

        new Thread(){
            @Override
            public void run() {
                while (true){
                    getlock2First();
                }
            }
        }.start();
    }
}

死鎖:

3、死鎖的排查

先進入jdk安裝的bin目錄,啟動命令行,輸入命令jps查看運行的線程ID

死鎖檢測1:

執行jstack -l 線程ID命令

死鎖檢測2:

可以看到Thread-1Thread-2都在相互等待對方持有的對象的monitor鎖釋放。

到此這篇關於Java多線程環境下死鎖模擬的文章就介紹到這瞭,更多相關Java多線程死鎖模擬內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: