深入理解JVM------分析死锁

前言

死锁,百度百科的解释是:死锁是指两个或两个以上的进程在执行过程中,由于竞争资源或者由于彼此通信而造成的一种阻塞的现象,若无外力作用,它们都将无法推进下去。

1、模拟死锁

public class DeadLockTest {

    public static void main(String[] args){
        A a = new A();
        B b = new B();
        a.next = b;
        b.next = a;
        new Thread(a,"aThread").start();
        new Thread(b,"bThread").start();
    }
}


class A extends T implements Runnable {
    T next;
    public synchronized void invoke() {
        System.out.println("当前线程:" + Thread.currentThread().getName()
                + " | 进入了" + this.getClass().getSimpleName() + " 获取到资源 | 准备调用:"
                + next.getClass().getSimpleName());
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        next.invoke();
    }
    @Override
    public void run() {
        invoke();
    }

}

class B extends T implements Runnable {
    T next;
    public synchronized void invoke() {
        System.out.println("当前线程:" + Thread.currentThread().getName()
                + " | 进入了" + this.getClass().getSimpleName() + " 获取到资源 | 准备调用:"
                + next.getClass().getSimpleName());
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        next.invoke();
    }

    @Override
    public void run() {
        invoke();
    }

}

class T {
    public synchronized void invoke(){ }
}

上面的代码也很简单,就是A对象持有B对象的引用,B对象持有A对象的引用。在线程a进入到invoke方法时锁住了A对象,线程b也进入到invoke方法锁住了B对象,线程a再调用B对象的invoke方法去试图锁住B对象时,发现B对象已经被锁住了,需要等待,同理b线程也是如此,就导致了两个线程相互等待对方释放锁。有点绕,如果不明白,好好品下。

咱们运行下上面的代码,看下结果:

当前线程:aThread | 进入了A 获取到资源 | 准备调用:B
当前线程:bThread | 进入了B 获取到资源 | 准备调用:A

2、查找进程

我们使用jps查找进程,再用jstack dump线程日志信息
在这里插入图片描述
dump的时候忘记加磁盘标识,结果dump的文件到C:\Users\Administrator这个目录里面去了,

3、分析

下面我们看看日志里面是怎么记录的,首先,我们全文搜索【deadlock】关键字,如下图:
在这里插入图片描述
上图展示了bThread和aThread发生了死锁,具体是哪块的代码还不清楚,咱们根据A对象0x000000078bb84c68继续往下搜索
在这里插入图片描述
现在咱们就很清楚死锁的代码所在了,剩下的就是优化代码了。

4、总结

一般来说,线程的日志是非常大,不可能一眼就能看出问题的所在,现在我们重点来说排查的思路:
首先就要找关键字,比如:
deadlock:表示有死锁。
Waiting on condition:等待某个资源或条件发生来唤醒自己。
Blocked:阻塞。
Waiting on monitor entry:在等待获取锁。

找到这些关键字,再找到具体的线程进行分析,最后根据线程死锁的执行的方法,去查看代码。

结束语

本篇到此把jstack命令介绍完了,下一篇将使用jmap的分析内存泄漏。

原创文章 55 获赞 76 访问量 17万+

猜你喜欢

转载自blog.csdn.net/cool_summer_moon/article/details/105643109