JVM問題排查——死鎖

在這裏插入圖片描述


環境

JDK8

 

準備工作

先準備好一個死鎖程序:(程序是網上隨便找的)

public class DeadLock {
    public static String obj1 = "obj1";
    public static String obj2 = "obj2";
    public static void main(String[] args){
        Thread a = new Thread(new Lock1());
        Thread b = new Thread(new Lock2());
        a.start();
        b.start();
    }    
}
class Lock1 implements Runnable{
    @Override
    public void run(){
        try{
            System.out.println("Lock1 running");
            while(true){
                synchronized(DeadLock.obj1){
                    System.out.println("Lock1 lock obj1");
                    Thread.sleep(3000);//獲取obj1後先等一會兒,讓Lock2有足夠的時間鎖住obj2
                    synchronized(DeadLock.obj2){
                        System.out.println("Lock1 lock obj2");
                    }
                }
            }
        }catch(Exception e){
            e.printStackTrace();
        }
    }
}
class Lock2 implements Runnable{
    @Override
    public void run(){
        try{
            System.out.println("Lock2 running");
            while(true){
                synchronized(DeadLock.obj2){
                    System.out.println("Lock2 lock obj2");
                    Thread.sleep(3000);
                    synchronized(DeadLock.obj1){
                        System.out.println("Lock2 lock obj1");
                    }
                }
            }
        }catch(Exception e){
            e.printStackTrace();
        }
    }
}

運行起來,會顯示如下:

Lock2 running
Lock2 lock obj2
Lock1 running
Lock1 lock obj1

此時已發生死鎖。針對兩個問題,我們開始排查:

  1. 哪幾個線程發生了死鎖
  2. 死鎖發生在哪幾行代碼
     

排查

姿勢一:使用JVisualVM排查

JDK8提供了一個可視化工具Jvisualvm,在jdk的bin目錄下可以找到。
雙擊打開它:
在這裏插入圖片描述從上圖可以看到這裏我們的項目DeadLock正在運行中。雙擊打開。
 
在這裏插入圖片描述
點擊“線程”標籤,可以看到上圖。
非常醒目的“檢測到死鎖!”紅色字樣。並且可以看到下面的線程圖中,Thread0和Thread1兩眼的紅色進度條,說明已經發生阻塞。
接下來查一下到底是哪一行代碼出現了問題:
單擊上圖中右上角的一個按鈕“線程Dump”。
 
此時會看到長段文字,一直往下翻,可以看到:
在這裏插入圖片描述
可以看到死鎖發生的位置。有了這些信息,我們就可以去review我們的代碼了。
 
 

姿勢二:使用Jps+Jstack

不熟悉這些常用的JVM調優工具?傳送門:JVM——命令行工具

首先打開我們的cmd,輸入jps:
在這裏插入圖片描述
通過jps獲取到我們的進程id是21036,然後利用jstack看看當前的線程快照:
輸入以下:

jstack -21036

會看到控制檯輸出了很多信息,直接翻到最後:
在這裏插入圖片描述到這裏就成功定位到了死鎖發生的代碼行。
 


除了上述的兩種方式,其實還有很多其他辦法,不過大體都是利用一些監控工具。
很多商用的監控工具功能非常全面且強大,例如Jprofiler等,這裏就不多介紹了。

 
更多資料歡迎關注:
在這裏插入圖片描述

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章