死锁:多个线程分别占用对方需要的同步资源不放弃,都在等待对方放弃。
死锁案例
以下案例演示了线程T1,T2出现死锁的情况:
线程T1在获得锁1的情况下还想获取线程T2拥有的锁2,而线程T2在拥有锁2的情况下还想获取线程1拥有的锁1。
public class DeadLockDemo {
public static void main(String[] args) {
Object obj1=new Object();
Object obj2=new Object();
new Thread(()->{
synchronized (obj1){
System.out.println("线程1获取锁1并尝试获取锁2");
try {
Thread.sleep(1000);
synchronized (obj2){
System.out.println("线程2获取锁2");
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},"T1").start();
new Thread(()->{
synchronized (obj2){
try {
System.out.println("线程2获取锁2并尝试获取锁1");
Thread.sleep(100);
synchronized (obj1){
System.out.println("线程2尝试获取锁1");
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},"T2").start();
}
}
定位死锁
当需要检测当前项目是否出现死锁并定位死锁出现的位置时,可以利用jconsole工具或者使用jps定位进程id,在用jstack定位死锁。
利用jps和jstack定位死锁
- 在IDEA的Terminal窗口先定位到当前项目路径下
- 再输入指令jps查看所有线程ID,接着用jstack ID 定位到需要查看的线程
- 最后根据输出内容可以得出当前线程是否出现死锁。
jconsole
- 搜索jconsole,运行。
- 连接到相应进程
- 检测死锁,查看结果
根据结果可以得到死锁发生在线程T1和T2。