线程死锁是指两个或两个以上的线程互相持有对方所需要的资源,由于synchronized的特性,一个线程持有一个资源,或者说获得一个锁,在该线程释放这个锁之前,其它线程是获取不到这个锁的,而且会一直死等下去,因此这便造成了死锁。
死锁产生的条件 互斥条件:一个资源,或者说一个锁只能被一个线程所占用,当一个线程首先获取到这个锁之后,在该线程释放这个锁之前,其它线程均是无法获取到这个锁的。占有且等待:一个线程已经获取到一个锁,再获取另一个锁的过程中,即使获取不到也不会释放已经获得的锁。不可剥夺条件:任何一个线程都无法强制获取别的线程已经占有的锁循环等待条件:线程A拿着线程B的锁,线程B拿着线程A的锁。。 死锁演示package com.github.excellent01; /** * @auther plg * @date 2019/5/18 12:12 */ public class DeadLock { private Object lock1 = new Object(); private Object lock2 = new Object(); public void method1() throws InterruptedException { synchronized(lock1){ System.out.println(Thread.currentThread().getName() + "获取到lock1,请求获取lock2...."); Thread.sleep(1000); synchronized (lock2){ System.out.println("获取到lock2...."); } } } public void method2() throws InterruptedException { synchronized(lock2){ System.out.println(Thread.currentThread().getName() + "获取到lock2,请求获取lock1...."); Thread.sleep(1000); synchronized (lock1){ System.out.println("获取到lock1...."); } } } public static void main(String[] args) { DeadLock deadLock = new DeadLock(); new Thread(()-> { try { deadLock.method1(); } catch (InterruptedException e) { e.printStackTrace(); } }).start(); new Thread(()-> { try { deadLock.method2(); } catch (InterruptedException e) { e.printStackTrace(); } }).start(); } }
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051运行结果:
这样便造成了死锁,Thread1拿着lock1,想要lock2,Thread2拿着lock2,想要lock1.
在发生死锁之后,程序就卡住了没有任何反应,但程序仍在运行,因此需要借助一些
首先使用 jps -l 显示正在运行的虚拟机进程,并显示虚拟机执行主类(main函数所在的类)名称以及这些进程的本地虚拟机唯一ID
死锁的原因,具体哪个线程,具体的行号都说明的很清楚。