线程死锁及其定位分析

死锁概述
线程死锁是指两个或两个以上的线程互相持有对方所需要的资源,由于synchronized的特性,一个线程持有一个资源,或者说获得一个锁,在该线程释放这个锁之前,其它线程是获取不到这个锁的,而且会一直死等下去,因此这便造成了死锁。

死锁产生的条件

  • 互斥条件:一个资源,或者说一个锁只能被一个线程所占用,当一个线程首先获取到这个锁之后,在该线程释放这个锁之前,其它线程均是无法获取到这个锁的。
  • 占有且等待:一个线程已经获取到一个锁,再获取另一个锁的过程中,即使获取不到也不会释放已经获得的锁。
  • 不可剥夺条件:任何一个线程都无法强制获取别的线程已经占有的锁
  • 循环等待条件:线程A拿着线程B的锁,线程B拿着线程A的锁。。

线程死锁代码示例:

/**
 * 死锁
 */
public class DeadLockTest {
    private String a;
    private String b;

    public DeadLockTest(String a, String b) {
        this.a = a;
        this.b = b;
    }

    public void m1() {
        synchronized (a) {
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "线程获得锁" + a + " 尝试获取锁" + b);
            synchronized (b) {
                System.out.println(Thread.currentThread().getName() + "线程获得锁" + b);
            }
        }
    }

    public static void main(String[] args) {
        DeadLockTest deadLock1 = new DeadLockTest("a", "b");
        DeadLockTest deadLock2 = new DeadLockTest("b", "a");

        new Thread(new Runnable() {
            @Override
            public void run() {
                deadLock1.m1();
            }
        }, "t1").start();

        new Thread(new Runnable() {
            @Override
            public void run() {
                deadLock2.m1();
            }
        }, "t2").start();
    }
}

运行结果:

线程死锁及其定位分析_第1张图片
看运行结果,如果没有外部条件中断,程序会 一直僵死在这

死锁的定位分析:
如果我们的程序出现了死锁,那么我们如何排查,定位呢?

在发生死锁之后,程序就卡住了没有任何反应,但程序仍在运行,因此需要借助一些命令来排查

  1. java的 jps -l : 显示正在运行的虚拟机进程,并显示虚拟机执行主类(main函数所在的类)名称以及这些进程的本地虚拟机唯一ID,如图:
    线程死锁及其定位分析_第2张图片

    可以看出10352 com.fangyajun.javasduty.juc.lock.DeadLockTest这个进程的类出现了问题。其余的是java自带的运行程序,不用管。

  2. 使用 jstack + 进程id 进行跟踪排查, 使用jstack命令可以打印出堆栈信息,如图:

    线程死锁及其定位分析_第3张图片

如上截图信息,看t2和t1信息,可以很明显的看出,是死锁导致的,原因找到了我们可以很容易的解决此次问题。

你可能感兴趣的:(JUC并发)