线程死锁检测与 分析工具深度解析
jconsole的使用:
产生死锁的实例:
public class MyTest3 {
public static void main(String[] args) {
//当A线程执行方法A.method的时候,会去执行B.method();
//这个时候,A尝试去获取B的锁,但是B的锁已经被B线程获取,此时A线程就会等待B线程释放B锁
//B线程与之相反,最终造成死锁。
new Thread(()->A.method(),"Thread-A").start();
new Thread(()->B.method(),"Thread-B").start();
}
}
class A {
//当synchronized被static修饰的时候,synchronized所持有的锁不是当前类的锁
//而是当前类A所对应的class对象所对应的锁,
//那么,这样使得不论A有多少个实例、对象,它们所拥有的都是同一把锁,都是类A所对应的class对象所拥有的锁
public static synchronized void method(){
System.out.println("method from A");
try {
Thread.sleep(3000);
B.method();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
class B {
public static synchronized void method(){
System.out.println("method from B");
try {
Thread.sleep(3000);
A.method();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
运行结果:死锁,两个线程一直在相互等待这对方锁持有的锁的释放
名称: Thread-A
//状态,说明线程B已经持有了java.lang.Class@55ecfe09对象的锁,所以A线程处于阻塞状态
状态: java.lang.Class@55ecfe09上的BLOCKED, 拥有者: Thread-B
//总阻止数:它阻止第二个线程获取这个锁;总等待数:它等待的锁的个数是1
总阻止数: 2, 总等待数: 1
堆栈跟踪:
jvm.memory.B.method(MyTest3.java:29)
jvm.memory.A.method(MyTest3.java:21)
- 已锁定 java.lang.Class@76edcb81 //已锁定,表示当前的A线程已经锁定ClassA所对应的class对象的锁
jvm.memory.MyTest3.lambda$main$0(MyTest3.java:8)
jvm.memory.MyTest3$$Lambda$1/1747585824.run(Unknown Source)
java.lang.Thread.run(Thread.java:748)
名称: Thread-B
...... 与A线程相反!