【多线程】死锁

news2025/1/17 4:00:35

🥰🥰🥰来都来了,不妨点个关注叭!
👉博客主页:欢迎各位大佬!👈

在这里插入图片描述

文章目录

  • 1. 死锁的三种情况
    • 1.1 一个线程一把锁(同一个线程给同一个对象加两次锁的情况)
    • 1.2 两个线程两把锁
    • 1.3 N个线程M把锁
  • 2. 造成死锁的 4 个必要条件
  • 3. 如何避免死锁?
    • 3.1 加锁顺序
    • 3.2 资源分配策略
    • 3.3 设置锁的超时机制
    • 3.4 死锁检测与恢复
    • 3.5 避免共享资源

在 常见的锁策略 这期内容中,介绍了几种常见的锁策略,其中,提到了"死锁"这一概念,本期内容具体讲解死锁~

1. 死锁的三种情况

1.1 一个线程一把锁(同一个线程给同一个对象加两次锁的情况)

一个线程一把锁的这个情况下,可重入锁没事,不可重入锁发生死锁!

class BlockingQueue {
    synchronized void put(int elem) {
        this.size();
        ...
    }
    
    synchronized int size() {
        ...
    }
}

在上述代码中,put()方法和size()方法这两个方法都给同一个锁对象 this 加锁,通过上述代码,可以看到,在 put()方法中调用了size()方法,进入put()方法的时候,对 this 对象加锁了,而在put()中调用size()时,又对 this 对象加了锁,这在日常开发中,很常见,实际上,这个情况并不会造成死锁,因为 synchronized 是可重入锁

常见的锁策略 介绍了可重入锁和不可重入锁,在这里,再简单介绍一下:

  • 可重入锁:允许同一个线程多次获取同一把锁,如果一个锁,在一个线程中,连续对锁,加锁两次,不死锁,即可重入锁
  • 不可重入锁:如果一个锁,在一个线程中连续对锁,加锁两次,死锁,则是不可重入锁

1.2 两个线程两把锁

两个线程两把锁的这个情况下,即使是可重入锁,也可能会死锁,比如这个情况,如下图:

在这里插入图片描述
t1 线程和 t2 线程并发执行,t1 线程先对 locker1 加锁,t2 线程先对 locker2 加锁,t1 线程继续执行,此时又要对 locker2 加锁,但是必须等待 t2 线程先释放 locker2,t2 线程继续执行,又要对 locker1 加锁,但是必须等待 t1 线程先释放 locker1,这样一来,就发生了死锁的情况

举一个生活中的栗子,这就好比在疫情期间,你没戴口罩准备去药店买口罩的时候,药店却让你必须戴口罩才能进入药店,这样就形成一个死锁的情况~

1.3 N个线程M把锁

线程的数量和锁的数量变多了后,更加复杂之后,就更容易造成死锁的情况~

这就涉及到一个著名的问题 —— “哲学家就餐问题” :5根筷子分别放在5位哲学家两两之间,哲学家想要吃面条,必须同时拿到左边的一根筷子和右边的一根筷子,只有一根筷子无法吃面条

这五个哲学家,
1)随机的进行去吃面条(拿筷子)和思考人生(放下筷子)
2)如果哲学家想要拿筷子,被别人占用了,就会等待,等的过程中不会放下手中已经拿到的筷子(非常地固执~)

在这里插入图片描述
可以看到,如果此时五位哲学家同时拿起左手边的筷子,他们还想拿起右边的筷子,但是筷子已经被旁边的哲学家左手边拿起来了,只能进行阻塞等待,就死锁了,即一个线程如果想要两次加锁,已经加了一个,另一个被抢走了,它就会一直等待,同时,占用着第一次加的锁

2. 造成死锁的 4 个必要条件

  • 互斥使用:即一个线程拿到一把锁之后,这个资源被这个线程占有时,另一个线程不能使用【锁的基本特点】
  • 不可抢占:即一个线程拿到锁,只能自己主动释放,不能被其它线程强行占有,资源请求者不能强制从资源占有者手中夺取资源 (不能挖墙脚呀!)【锁的基本特点】
  • 请求和保持:即资源请求者在请求其它资源的同时,保持对原有资源的占有(典型的吃着碗里的看着锅里的)【代码的特点】
  • 循环等待:即存在一个等待队列,P1 占有 P2 资源,P2 占有 P3 资源,P3 占有 P4 资源,P4 资源占有 P1资源…,形成逻辑依赖循环的等待环路(钥匙锁车里了,车钥匙又锁家里了)(上述哲学家中,哲学家1等待5,2等待1,3等待2,4等待3,而5又等待1)【代码的特点】

3. 如何避免死锁?

避免死锁有很多种方法,这里介绍五种(重点介绍第一种:加锁顺序)

3.1 加锁顺序

死锁是一个比较严重的 BUG,实践中如何避免死锁呢?

当造成死锁的4个必要条件都成立时,形成死锁,在死锁的情况下,如果打破上述任何一个条件,死锁情况就可以避免,其中一个简单有效的办法是,破解循环等待这个条件

具体操作针对锁进行编号,如果需要同时获取多把锁,约定加锁顺序,务必是先对小的编号加锁,后对大的编号加锁

1)针对N个线程M把锁情况

约定:每个哲学家只能先获取左手和右手中编号较小的筷子,2 号哲学家先获取1 号筷子,剩下的哲学家也依次获取左右手之间编号较小的筷子,轮到最后一位 1 号哲学家的时候,由于 1 号筷子已经被占用,他就无法获取 1 号筷子,从而进入阻塞等待,这样哲学家还想拿一根筷子的时候,5 号哲学家可以拿到编号5的筷子,吃完面条后,将编号4和5的筷子放下,依次的3号、2号哲学家也能完成吃面的任务,等到 2 号哲学家放下筷子,1 号哲学家就可以拿到 1 号筷子和 5 号筷子,从而结束阻塞等待,不会形成死锁情况,如下图:

在这里插入图片描述
因此,只要约定了加锁顺序,循环等待条件就会自然破除,死锁的情况也就可以避免,体现在代码中,就是只要是一个线程中要加多把锁,就一定需要注意加锁顺序,可以约定每次加锁的时候都先给编号小的加锁,后给编号大的加锁,并且所有的线程都遵循这个顺序即可

2)针对两个线程两把锁情况

只要每次加锁的时候都先给 locker1 加锁,后给 locker2 加锁即可,将线程加锁顺序固定下来,可以破除循环等待,如下图所示:

在这里插入图片描述

3.2 资源分配策略

可以使用 银行家算法 等资源分配策略,主要用于多道程序系统中避免死锁的发生,银行家算法通过预测资源分配后的系统状态,来避免系统进入不安全状态,从而防止死锁的发生,其本质是对资源更合理的分配,但是本身这个算法比较复杂,实现这个算法本身还可能引入额外的 BUG,因此,不适合实际开发中使用(这里不详细展开)

3.3 设置锁的超时机制

在获取锁操作的过程中设置一个超时时间,在等待超过一定时间后放弃获取锁,并进行相应的处理,比如执行其它操作或重试,可避免长时间等待造成系统阻塞,从而避免死锁

3.4 死锁检测与恢复

在开发过程中,可以使用专门的工具,比如 Java 中 jstack,来检测死锁线程的状态和调用栈信息,通过周期性地检测系统中是否存在死锁,并采取相应的措施进行恢复,例如终止某些进程或回滚操作

3.5 避免共享资源

尽量减少进程间共享资源的数量,或者采用副本的方式而不是共享资源的方式,避免资源竞争导致死锁的可能性

💛💛💛本期内容回顾💛💛💛

在这里插入图片描述

✨✨✨本期内容到此结束啦~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2110258.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Save OpenAI response in Azure function to Blob storage

题意:将 OpenAI 的响应保存在 Azure 函数中到 Blob 存储 问题背景: I used blob trigger template to create an Azure function that is triggered by new file updated in Azure blob storage. I am using python v2 to create Azure function in VSc…

SAP ABAP任意表数据查询+快速下载工具

背景: 项目上业务顾问有时候需要下载标准表大量的数据到Excel进行一些比对,但是标准SE16N,SE16的下载电子表格功能在遇到大批量数据的时候会非常慢,于是抽空写了个通用的查询下载工具,可以快速的下载数据。 工具界面&…

Linux - Linux安装部署xxl-job

一、下载源码 xxl-job源码地址: https://github.com/xuxueli/xxl-job 2.4.2版本为例:https://github.com/xuxueli/xxl-job/archive/refs/tags/2.4.1.tar.gz xxl-job文档地址: 分布式任务调度平台XXL-JOB 二、安装依赖环境 因为需要…

Go语言中的RPC协议原理解析

Go语言中的RPC协议原理解析 在分布式系统中,不同的服务或组件通常运行在不同的计算机或进程上。为了实现这些服务之间的通信,我们可以使用RPC(Remote Procedure Call,远程过程调用)协议。RPC允许我们像调用本地函数一…

【区块链 + 人才服务】FISCO BCOS 高校实训和管理平台 | FISCO BCOS应用案例

深圳市火链文化传播有限公司建设部署 FISCO BCOS 高校实训和管理平台,这是一条多领域覆盖的联盟链,以高 校 政府教育部门 合作企业共同授权成为联盟链节点,实现在政府指导监管下,多行业扩展、多机构参与、多 平台共存的联盟链模…

SparseDrive - 清华地平线开源的e2e的框架

清华地平线合作开发的e2e的框架 SparseDrive资源 论文 https://arxiv.org/pdf/2405.19620 git https://github.com/swc-17/SparseDrive 个人觉得该文章厉害的地方 纯sparse mapping, 3d detection方案, 用的检测头sparse4D V3 sparsev1v2v3基本一致,m…

秒懂:父子进程

1.概念 在操作系统中,当一个进程(称为父进程)创建另一个新进程(称为子进程)时,父子进程之间建立了一种特殊的关系。 2.创建父子进程的方法 2.1 fork()函数详解: fork 是一个系统调用&#xff0…

Qt 加载 WPS 时提示要登录

项目中Qt加载word时 默认用wps打开word文档 程序一运行老是提示要立即登录 看着很烦 可以按下面的方法去掉这个烦人的东西 在下面的项目中新建字符串enableforceloginforfirstinstalldevice,值为false即可。

品聚文化--打造票圈神话

热烈祝贺鸿玉祥安文票9月5号正式启动运营 尊敬的各位领导、合作伙伴、客户以及全体品聚的家人们 今天,我们迎来了公司文票正式启动这一历史性的时刻,在此,我谨代表公司及全体员工,向所有关心和支持我们发展的朋友们表示最热烈的祝…

Java进阶13讲__第十讲

IO流、File 1. File 1.1 创建对象 1.1.1 File指向 指向文件 File f1 new File("E:\\TableFace\\Finished\\合同审核.txt"); 指向文件夹 File f2 new File("E:\\TableFace\\Finished"); 指向不存在文件 File f2 new File("E:\\T…

用最简单的话来解释大模型中的Transformer架构

开篇 我个人的观点是要想系统而又透彻地理解 Transformer,至少要遵循下面这样一个思路(步骤): 理解NLP基础 在探讨Transformers之前,了解自然语言处理(NLP)的一些基本知识至关重要。NLP使计算机能够理解和生成自然语…

GEE数据集:美国国家结构清单(NSI)用于评估和分析自然灾害的点式结构清单

简介 国家结构清单 国家结构清单(NSI)是一个数据库系统,其中包含不同质量和空间覆盖范围的结构清单。 NSI 数据库的目的是促进存储和共享用于评估和分析自然灾害的点式结构清单。 洪水风险是主要用途,但每个结构都有足够的数据来…

prompt实用技巧-竞对分析-飞书发布会上多维表和低代码平台原型分析

prompt engineer 工程师的一天 ,竞品分析相关。 “作为企业软件咨询公司整理出uipath,salesforce,airtable,zapier的官网地址,功能点详情,优势,劣势 生成表格4*4,表头为:uipath,salesforce,airtable,zapier, 第一例为&…

Nvidia股价反弹,博雅分析师力挺“行业首选”

在周四的交易中,Nvidia(NVDA)股价成功收复部分失地,上涨1%,这一积极走势得益于博雅全球研究公司(美国银行全球研究)发布的一份看涨报告。博雅分析师重申了对Nvidia的“行业首选”评级&#xff0…

『功能项目』Unity本地数据库读取进入游戏【29】

本章项目成果展示 打开上一篇28Unity连接读取本地数据库的项目, 本章要做的事情是通过读取本地数据库登录进入游戏场景 首先创建一个脚本文件夹: 新建脚本:MySqlAccess.cs 编写脚本:MySqlAccess.cs using UnityEngine; using MyS…

自定义v-model的两种形式

1 使用update (1)在组件中使用v-model"value" (2)props中接收modelValue (3)然后在value值发生变化的时候:this.$emit("update:modelValue", this.value); (4&…

【压力测试】如何确定系统最大并发用户数?

一、明确测试目的与了解需求 明确测试目的:首先需要明确测试的目的,即为什么要确定系统的最大并发用户数。这通常与业务需求、系统预期的最大用户负载以及系统的稳定性要求相关。 了解业务需求:深入了解系统的业务特性,包括用户行…

大语言模型(LLM)如何更好地继续预训练(Continue PreTraining)

预训练(Pretraining)是一个非常消耗资源的工作,尤其在 LLM 时代。随着LLama2的开源,越来越多人都开始尝试在这个强大的英文基座模型上进行中文增强。但,我们如何才能保证模型在既学到「中文知识」的情况下,…

这本PyTorch官方出品的《Deep Learning With PyTorch》终于有了中文版!

导读: 一些线上或线下的关于PyTorch的课程和大学里的计划课程,以及大量的线上博客和教程,使得PyTorch学习起来更容易。然而,关于PyTorch的图书很少。随着本书的出版,我们最终有了一本关于PyTorch的官方权威著作。 它非…

下载适用于 Linux 的 MongoDB Shell教程

下载适用于 Linux 的 MongoDB Shell教程 下载地址 MongoDB Shell Download | MongoDB 2、往下拉,选择Linux对应的版本下载