利用ThreadMXBean实现检测死锁

在使用JConsole的时候,在线程页下,可以看到一个检测死锁按钮,很好奇它是如何获取死锁线程的。同时检测死锁算法也是操作系统的重要算法之一,本文在参考了JConsole源码的基础上来实现检测死锁的功能。
获取源码
https://github.com/maoturing/bescode/tree/master/src/com/requirement/detectdeadlock

1.检测死锁类的属性和方法

public class DeadlockDetector{

  //获取ThreadMXBean 
  private final ThreadMXBean mBean = ManagementFactory.getThreadMXBean();

  //处理检测到的死锁线程
  public void handleDeadLock(ThreadInfo[] deadLockThreads);

  //得到线程锁定的对象
  public void getThreadLock(long selected);

  //得到线程和线程所在的组,K为线程Id,V为所在的组,组以死锁划分,一个死锁为一组
  public Map getDeadlockedGroup() throws IOException;

2.获取死锁线程

// 查找因为等待获得对象监视器或可拥有同步器而处于死锁状态的线程循环。返回线程ID
long[] ids = mBean.findDeadlockedThreads(); 

// 根据死锁线程id得到死锁线程信息,参数2代表的是获取同步监视器,后面获取线程锁定对象时使用
ThreadInfo[] threadInfos = mBean.getThreadInfo(ids , true, false); 

3.处理死锁线程

public void handleDeadLock(ThreadInfo[] deadLockThreads) {
        if (deadLockThreads != null) {
            System.err.println("Deadlock detected!");

            for (ThreadInfo threadInfo : deadLockThreads) {
                if (threadInfo != null) {
                    // SecurityException
                    for (Thread thread : Thread.getAllStackTraces().keySet()) {
                        if (thread.getId() == threadInfo.getThreadId()) {
                            System.out.println("id:" + threadInfo.getThreadId());
                            System.out.println("名称:" + threadInfo.getThreadName());
                            System.out.print("状态:" + threadInfo.getLockName());
                            System.out.println("上的" + threadInfo.getThreadState());
                            System.out.println("拥有者:" + threadInfo.getLockOwnerName());
                            System.out.println("总阻止数:" + threadInfo.getBlockedCount());
                            System.out.println("总等待数:" + threadInfo.getWaitedCount());
                            System.out.println("状态:" + threadInfo.toString());

                            String name = mBean.getThreadInfo(threadInfo.getLockOwnerId()).getLockName();
                            System.out.println("已锁定:" + name);
                            int i = 0;

                            MonitorInfo[] monitors = threadInfo.getLockedMonitors();

                            for (StackTraceElement ste : thread.getStackTrace()) {
                                System.err.println("堆栈深度:"+thread.getStackTrace().length);
                                System.err.println("堆栈信息:"+ste.toString());

                                System.out.println("拼接的堆栈信息:"+ste.getClassName() + ste.getMethodName() + ste.getFileName()
                                        + ste.getLineNumber());
                                selectThread(threadInfo.getThreadId());
                                            }
                        }

                    }
                    System.out.println("==================");
                }
            }

        } else {
            System.out.println("未检测到死锁线程");
        }

    }

4.得到线程锁定的对象
首先需要得到线程对象,mBean.getThreadInfo(deadlockedThreads, true, false)方法的第二个参数为true表示获得同步监视器,

public void getThreadLock(long selected) {
        final long threadID = selected;
        StringBuilder sb = new StringBuilder();
        ThreadInfo ti = null;
        MonitorInfo[] monitors = null;
            for (ThreadInfo info : infos) {
                if (info.getThreadId() == threadID) {
                    ti = info;
                    monitors = info.getLockedMonitors();
                    break;
                }
            }
            System.out.println("support");
        if (ti != null) {
            int index = 0;
            if (monitors != null) {
                for (MonitorInfo mi : monitors) {
                    if (mi.getLockedStackDepth() == index) {
                        System.out.println("已锁定:" + mi.toString());
                    }
                }
                index++;
            }
        }

    }

5.得到线程所在的组(检测死锁算法)
线程所在的组的区分标准是线程是否属于同一个死锁,所谓死锁是指两个或两个以上的线程,互相竞争资源导致的一种阻塞现象,最简单的例子就是线程 T1 已锁定B对象,然后去请求A对象,而线程 T2 已锁定A对象,然后去请求B对象,导致二者一直阻塞下去。

两个线程竞争资源导致的死锁.png

当然,实际可能出现的情况会更复杂,我们如何来判断哪些线程属于同一个死锁呢?
在下图中可以看出,无论是哪种死锁形式,都会形成一个闭环。根据这个特点,我们可以将死锁抽象为有向图,多个死锁就是多个有向图,我们改造一下有向图的遍历算法即可得到所有的死锁以及死锁拥有的线程。


三种死锁形式

首先遍历所有被阻塞的线程T1~T7,第一次先得到线程T1,由于T1被B对象阻塞,我们使用getLockOwnerId()方法获得B对象的拥有者线程T2,并将T1的访问标志visited[j]置为true,继续执行.......,直到得到了线程T3,用getLockOwnerId()方法获得线程T1,此时发现线程T1的访问标志visited[j]已经为true,意味着我们已经完成了一个死锁的遍历,之前遍历过的线程组成了1个死锁,此时标志死锁个数的gid++,继续遍历下一个死锁。

    public Map getDeadlockedGroup() throws IOException {

        long[] ids = mBean.findDeadlockedThreads();
        if (ids == null) {
            return null;
        }
        ThreadInfo[] infos = mBean.getThreadInfo(ids, Integer.MAX_VALUE);

        List<Long[]> dcycles = new ArrayList<Long[]>();
        List<Long> cycle = new ArrayList<Long>();
        Map<Long, Integer> map = new HashMap<Long, Integer>();
        int gid = 1;
        boolean[] visited = new boolean[ids.length];

        int index = -1; // Index into arrays
        while (true) {
            if (index < 0) {
                if (map.size() > 0) {
                    // a cycle found
                    // dcycles.add(cycle.toArray(new Long[0]));
                    gid++;
                    // cycle = new ArrayList<Long>();
                }
                // start a new cycle from a non-visited thread
                for (int j = 0; j < ids.length; j++) {
                    if (!visited[j]) {
                        index = j;
                        visited[j] = true;
                        break;
                    }
                }

                // 当所有线程均被访问过,退出while循环
                if (index < 0) {
                    // done
                    break;
                }
            }

            // cycle.add(ids[index]);
            map.put(ids[index], gid);
            long nextThreadId = infos[index].getLockOwnerId();

            for (int j = 0; j < ids.length; j++) {
                ThreadInfo ti = infos[j];
                if (ti.getThreadId() == nextThreadId) {
                    if (visited[j]) {
                        index = -1;
                    } else {
                        index = j;
                        visited[j] = true;
                    }
                    break;
                }
            }
        }
        //线程id为key,所在死锁组为value
        return map;
    }

对于死锁分组,由于遍历算法的原因,可能会出现一个线程组成的死锁的小bug,比如“图-三种死锁形式”中的第三种情况,如果从T8开始遍历,那么T8和T7组成一个死锁,T6单独组成一个死锁,这种情况极为少见,JConsole也是这样处理,所以这里就不深入追究了。以上,就是java利用ThreadMXBean获取线程死锁的所有内容。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容

  • Java8张图 11、字符串不变性 12、equals()方法、hashCode()方法的区别 13、...
    Miley_MOJIE阅读 3,698评论 0 11
  • Java SE 基础: 封装、继承、多态 封装: 概念:就是把对象的属性和操作(或服务)结合为一个独立的整体,并尽...
    Jayden_Cao阅读 2,105评论 0 8
  • Java-Review-Note——4.多线程 标签: JavaStudy PS:本来是分开三篇的,后来想想还是整...
    coder_pig阅读 1,643评论 2 17
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,221评论 11 349
  • 刚开始被男神拉进群里的时候挺意外的,因为不觉得自己做得好,尤其是在目标的实现上,感觉自己这一期相比较上一期,除了保...
    一只特栗独行阅读 389评论 0 0