1.活跃性问题
大白话解释:一个线程内的代码本来是有限的,但是由于某种原因,此线程中的代码一直执行不完,这就叫线程的活跃性问题。线程活跃性分为如下三种现象:
(1)死锁
(2)活锁
(3)饥饿
此篇文章给出一个死锁示例和死锁的排查方法。
2.死锁示例
2.1 代码示例
@Slf4j(topic = "c.DeadLockDemo")
public class DeadLockDemo {
public static void main(String[] args) {
Object A = new Object();
Object B = new Object();
Thread t1 = new Thread(() -> {
synchronized (A) {
log.debug("lock A");
Sleeper.sleep(1); // 此处睡一秒,让线程t2有机会获取到B锁
synchronized (B) {
log.debug("lock B");
log.debug("操作...");
}
}
}, "t1");
Thread t2 = new Thread(() -> {
synchronized (B) {
log.debug("lock B");
Sleeper.sleep(1); // 此处睡一秒,让线程t1有机会获取到A锁
synchronized (A) {
log.debug("lock A");
log.debug("操作...");
}
}
}, "t2");
t1.start();
t2.start();
}
}
2.2. 运行代码,出现死锁现象
运行上述代码,出现死锁现象如下图1所示。
图 1 运行程序,出现死锁
3.死锁的检测方法
3.1. jps和jstack命令配合查看死锁问题
3.1. jps命令
通过 【jps -l】命令查看具体的进程号,如下图2所示。
图 2 jps命令查看java程序进程号
3.2 jstack命令
通过【jstack -l 进程号】查看死否有死锁问题,如下图3和图4所示。
图 3 两个进程处于阻塞(BLOCKED)状态
图 4 死锁的详细描述信息