前言
在 Java 的并发编程中,有一个问题需要特别注意,那就是死锁,如果发生了死锁,基本就是重启,而重启将会丢失运行中的数据。所以,了解死锁的形成并排查死锁到预防死锁成了一个重要的问题。
我们了解任何一个事情的步骤是:what,how,why,why not。
1. 什么是死锁?
我们还是直接写一段代码来看看:
package hello;
public class DeadLock {
public static void main(String[] args) {
new Thread(() -> {
try {
new DeadLock().resource1();
} catch (InterruptedException e) {
}
}
).start();
new Thread(() -> {
try {
new DeadLock().resource2();
} catch (InterruptedException e) {
}
}
).start();
}
void resource1() throws InterruptedException {
synchronized ("resource1") {
System.out.println("获取资源1");
// 等待 1 秒让另一个线程拿到锁
Thread.sleep(1000);
resource2();
}
}
void resource2() throws InterruptedException {
synchronized ("resource2") {
System.out.println("获取资源2");
// 等待 1 秒让另一个线程拿到锁
Thread.sleep(1000);
resource1();
}
}
}
上面的代码中,我们启用了两个线程,分别抢占2个资源,但这两个资源又分别被不同的对象(字符串)锁住了。当第一个线程调用 resource1 方法,进入同步块,拿到锁,并等待 1 秒钟让另一个线程进入 resource2 同步块,当第二个线程进入同步块后,注意:此时, 拿着 resourec1 锁的线程企图拿到 resource2 的锁,但这个时候,拿着 resource2 的线程也想去拿 resource1 的锁。于是就出现了互相僵持的情况,谁也无法拿到对方的锁,整个系统就卡死了。
这种情况就是死锁。
像我们现在写的代码是自己故意造出来的死锁,我们能够发现,那如果是线上环境怎么办,假如我们的系统卡死了,我们怎么知道到底是哪一段代码出现了问题,有没有可能使死锁的问题。也就是如何检测死锁。
2. 如何检测死锁?
由于死锁极难通过人工的方式查出来,因此JDK 提供了命令来检测某个java进程中心线程的情况,并排查有没有死锁。上面命令呢? jps , 用来查看java 程序的进程号,当然在 Linux 中也可以通过别的方式获取, jstack 进程号
命令则可以答应对应进程的栈信息,并找到死锁。
我们就刚刚的程序,在 windows 上使用该命令。
C:\Users\stateis0>jps
11060
2084 Launcher
10712 RemoteMavenServer
18040 Jps
11820 DeadLock
C:\Users\stateis0>jstack 11820<