死锁的产生与解决

死锁产生的原因

产生死锁的四个必要条件:

  • 互斥使⽤: 即当资源被⼀个线程使⽤(占有)时,别的线程不能使⽤
  • 不可抢占: 资源请求者不能强制从资源占有者⼿中夺取资源,资源只能由资源占有者主动释放。
  • 请求和保持: 即当资源请求者在请求其他的资源的同时保持对原有资源的占有。
  • 循环等待: 即存在⼀个等待队列:P1占有P2的资源,P2占有P3的资源,P3占有P1的资源。这样就形成了⼀个等待环路。
java 复制代码
Object lock1 = new Object();
Object lock2 = new Object();

Thread t1 = new Thread() {
 @Override
 public void run() {
     synchronized (lock1) {
         synchronized (lock2) {
 // 执行逻辑
      }
    }
  }
};
t1.start();

Thread t2 = new Thread() {
 @Override
 public void run() {
     synchronized (lock2) {
         synchronized (lock1) {
 // 执行逻辑
       }
    }
  }
};
t2.start();

这是一个死锁的代码,可以看到t1先获取的lock1锁再获取lock2锁. 而t2恰好相反. 当t1获取lock1时想要接着去获取lock2时, 此时t2已经先获取到lock2了, 此时就会进入冲突, t1占用lock1需要lock2, t2占用lock2却需要lock1.

需要注意的是 死锁产生必须四个条件都具备,而破坏死锁只需要破坏其中一个条件即可

死锁的解决办法

针对死锁产生的原因是: 因为线程之间获取锁的顺序不同,导致形成环路. 所以解决办法就是破坏这种环路;

对锁进行标号排序:

将所有的锁标上一个序号, 获取锁的顺序要根据锁的标号进行, 并且所有线程都需要根据这个原则来进行锁的获取

java 复制代码
Object lock1 = new Object();
Object lock2 = new Object();
Object lock3 = new Object();

Thread t1 = new Thread() {
 @Override
 public void run() {
     synchronized (lock1) {
         synchronized (lock2) {
            synchronized(lock3){
             // 执行逻辑
          }
       }
     }
  }
};
t1.start();

Thread t2 = new Thread() {
 @Override
 public void run() {
     synchronized (lock1) {
         synchronized (lock2) {
            synchronized(lock3){
                 // 执行逻辑
            }
        }
     }
  }
};
t2.start();

例如将所有锁取名时加上顺序1,2,3. 然后获取锁的时后遵从从1,2,3开始获取锁, 避免了环路问题.

相关推荐
Highcharts.js1 天前
倒置百分比堆叠面积图表示列详解|Highcharts大气成分图表代码
开发语言·信息可视化·highcharts·图表开发·面积图·图表示例·推叠图
csdn_aspnet1 天前
C语言 Lomuto分区算法(Lomuto Partition Algorithm)
c语言·开发语言·算法
Dicky-_-zhang1 天前
消息队列Kafka/RocketMQ选型与高可用架构:从单体到100万TPS的演进
java·jvm
晨曦中的暮雨1 天前
4.15腾讯 CSIG云服务产线 一面
java·开发语言
存在morning1 天前
【GO语言开发实践】二 GO 并发快速上手
大数据·开发语言·golang
fake_ss1981 天前
AI时代学习全栈项目开发的新范式
java·人工智能·学习·架构·个人开发·学习方法
茉莉玫瑰花茶1 天前
工作流的常见模式 [ 1 ]
java·服务器·前端
未若君雅裁1 天前
Spring AOP、日志切面与声明式事务原理
java·后端·spring
No8g攻城狮1 天前
【人大金仓】wsl2+ubuntu22.04安装人大金仓数据库V9
java·数据库·spring boot·非关系型数据库
xiaoerbuyu12331 天前
开源Java 邮箱 基于SpringBoot+Vue前后端分离的电子邮件
java·开发语言