一. 死锁
- 什么是死锁?
- 死锁说白了,就是在我们的程序中出现了锁的嵌套,外面一个锁,里面一个锁,这就形成了死锁!
- 死锁它不是一个知识点,它是一个错误,现在学习死锁的目的就是以后不要去犯这个错误!
- A线程拿着A锁,B线程拿着B锁,它们都在等着对方释放锁,所以这个时候程序就会卡死,运行不下去了!
- 在以后写锁的时候,千万千万不要让两个锁嵌套起来!
- 死锁就是多个线程互相持有对方已经获取到的锁,进行互相等待,就导致死锁的产生。
- 死锁原因:不同线程分别占用对方需要的资源不放弃,都在等待对方先放弃自己所需要的同步资源,就形成了死锁。
- 出现死锁不会出现异常,也不会出现提示,所有线程都处于阻塞状态,无法继续。
- 比如两个人去开车,一个人有钥匙,另一个人有车,有车的等对方把钥匙给他,有钥匙的等对方把车给他,两个人谁也不让,就只能互相等着。
概述:
线程死锁是指由于两个或者多个线程互相持有对方所需要的资源,导致这些线程处于等待状态,无法前往执行。
什么情况下会产生死锁
资源有限
同步嵌套
package com.gch.d10_dead_lock;
public class MyThread extends Thread {
/**
* 调用父类的有参构造器
* @param name:线程名
*/
public MyThread(String name){
super(name);
}
// 定义A锁、B锁
public static Object objA = new Object();
public static Object objB = new Object();
@Override
public void run() {
// 1.循环
while(true){
if("线程A".equals(getName())){
synchronized(objA){
System.out.println("线程A拿到了A锁,准备拿B锁");
}
synchronized(objB){
System.out.println("线程A拿到了B锁,顺利执行完下一轮");
}
}else if("线程B".equals(getName())){
if("线程B".equals(getName())){
synchronized(objB){
System.out.println("线程B拿到了B锁,准备拿A锁");
}
synchronized(objA){
System.out.println("线程B拿到了A锁,顺利执行完下一轮");
}
}
}
}
}
}
package com.gch.d10_dead_lock;
public class ThreadDemo {
public static void main(String[] args) {
/*
需求:死锁
*/
// 1.创建线程对象
Thread t1 = new MyThread("线程A");
Thread t2 = new MyThread("线程B");
// 2.启动线程
t1.start();
t2.start();
}
}
如何避免线程死锁?
- 1. 避免一个线程同时获得多个锁
- 2. 避免一个线程在锁内同时占用多个资源,尽量保证每个锁只占用一个资源
- 3. 尝试使用定时锁,使用lock.tryLock(timeout)来替代使用内部锁机制