什么是死锁?
死锁指的是某个资源占用后,一直得不到释放,导致其他需要这个资源的线程进入阻塞状态。
产生死锁的4个必要条件
- 互斥条件:在一段时间内某资源仅为一个线程所占有。
- 不可剥夺条件:线程所获得的资源在未使用完毕之前,不能被其他线程强行夺走。
- 请求和保持条件:线程已经保持了至少一个资源,但又提出了新的资源请求,而该资源已被其他线程占有。
- 循环等待条件:存在一种线程资源的循环等待链,链中每一个线程已获得的资源同时被链中下一个线程所请求。
public class DeadClockDemo {
private static final Object HAIR_A = new Object();
private static final Object HAIR_B = new Object();
public static void main(String[] args) {
new Thread(()->{
synchronized (HAIR_A){
try {
Thread.sleep(50L);
}catch (Exception e){
e.printStackTrace();
}
synchronized (HAIR_B) {
System.out.println("hairA抓到了hairB的头发");
}
}
}).start();
new Thread(()->{
synchronized (HAIR_B){
synchronized (HAIR_A){
System.out.println("hairB抓到了hairA的头发");
}
}
}).start();
}
模拟死锁的场景,控制台不输出,两个线程处于僵持的状态,死锁产生
查看死锁产生的方法
cmd 命令 -> 输入jps 可以查询到对应demo的pid ,图中我做的DeadClockDemo 相对应的Pid是9536
再用jstack命令查看整个demo的运行情况
可以看到线程的执行情况,并检测到有死锁的产生
也可使用jdk自带的jconsole监控来查看
cmd 命令jconsole,选择死锁的项目demo连接后选择线程,可以看到检测死锁的按钮
可以看到相应死锁的堆栈信息。