【多线程】死锁

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

在这里插入图片描述

文章目录

  • 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.mzph.cn/pingmian/53217.shtml

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

相关文章

彻底解决 node/npm, Electron下载失败相关问题, 从底层源码详解node electron 加速配置

最近玩了一下electron项目, 总是会遇到electron的下载失败问题, 于是看了一下node源码, 做一个记录. node/npm 加速配置 这个配置通过设置node配置里面的registry 这个配置项来完成加速. 配置方法 npm config set registry https://registry.npmmirror.com上面的命令就是将当…

【全网最全】2024年数学建模国赛C题超详细保奖思路+可视化图表+成品论文+matlab/python代码等(后续会更新

您的点赞收藏是我继续更新的最大动力! 一定要点击如下的卡片,那是获取资料的入口! 基于优化模型的农作物的种植策略 摘要 随着农业生产向集约化和智能化方向发展,优化种植策略以最大化经济收益成为当前农业研究中的重要问题。本…

车载测试协议:ISO-14229、ISO-15765、ISO-11898、ISO-26262【车企项目实操学习】②

FOTA模块中OTA的知识点:1.测试过程中发现哪几类问题? 可能就是一个单键的ecu,比如升了一个门的ecu,他的升了之后就关不上,还有就是升级组合ecu的时候,c屏上不显示进度条。 2.在做ota测试的过程中&#xff…

已入职华为!!关于我成功拿下华为大模型算法岗经验总结

方向:大模型算法工程师 整个面试持续了1小时10分钟,能够看出面试官是典型搞技术的,问的很专业又很细,全程感觉压力好大,面完后感觉丝丝凉意,不过幸好还是成功拿下了Offer 一面: 自我介绍 简历项目深度交流 1.项目的背…

Java笔试面试题AI答之JDBC(2)

文章目录 7. 列出Java应该遵循的JDBC最佳实践?8. Statement与PreparedStatement的区别,什么是SQL注入,如何防止SQL注入Statement与PreparedStatement的区别什么是SQL注入如何防止SQL注入 9. JDBC如何连接数据库?1. 加载JDBC驱动程序2. 建立数…

[网络原理]关于网络的基本概念 及 协议

文章目录 一. 关于网络的概念介绍1. 局域⽹LAN2. ⼴域⽹WAN3. 主机4. 路由器5. 交换机IP地址端口号 二. 协议协议分层TCP/IP五层模型(或四层)OSI七层模型封装分用 一. 关于网络的概念介绍 1. 局域⽹LAN 局域⽹,即 Local Area Network,简称LAN。 Local …

二叉树的层次遍历(10道)

&#xff08;写给未来遗忘的自己&#xff09; 102.二叉数的层序遍历&#xff08;从上到下&#xff09; 题目&#xff1a; 代码&#xff1a; class Solution { public: vector<vector<int>> levelOrder(TreeNode* root) { vector<vector<int>> r…

JVM系列(十) -垃圾收集器介绍

一、摘要 在之前的几篇文章中,我们介绍了 JVM 内部布局、对象的创建过程、运行期的相关优化手段以及垃圾对象的回收算法等相关知识。 今天通过这篇文章,结合之前的知识,我们一起来了解一下 JVM 中的垃圾收集器。 二、垃圾收集器 如果说收集算法是内存回收的方法论,那么…

稀土废水回收硫酸铵树脂技术

稀土废水回收硫酸铵的过程主要涉及到化学沉淀法、离子交换法和蒸发结晶法等技术。这些方法可以有效地从稀土废水中回收硫酸铵&#xff0c;同时降低废水中的氨氮含量&#xff0c;实现资源的循环利用。以下是具体的技术介绍&#xff1a; 稀土废水回收硫酸铵的技术 ● 化学沉淀…

【MATLAB】矩阵的合并

矩阵的合并是指将两个或者多个矩阵合并到一起构成一个新的矩阵。矩阵标识符方括号 [ ]&#xff0c;不仅可以用来创建新的矩阵&#xff0c;还可以用来将若干个矩阵合并到一起。表达式 C [A B] 将矩阵A和B在水平方向上合并到一起&#xff0c;而表达式C[A;B]则将矩阵A和B在竖直方…

java项目docker部署时进行热部署

本文需要pontwiner进行配合操作 1.上传文件到对应服务器&#xff0c;可以通过xftp等文件上传工具进行文件上传 2.获取docker imagId XX为项目部署名称 例如&#xff1a;test-server docker ps -a |grep XX 3.复制文件到docker容器的/tmp目录下 docker cp XXXX.class im…

WEB服务与虚拟主机/IIS中间件部署

WWW&#xff08;庞大的信息系统&#xff09;是基于客户机/服务器⽅式的信息发现技术和超⽂本技术的综合。网页浏览器//网页服务器 WWW的构建基于三项核⼼技术&#xff1a; HTTP&#xff1a;超文本传输协议&#xff0c;⽤于在Web服务器和客户端之间传输数据。HTML&#xff1a;⽤…

工业制造企业如何与供应商间 进行高效安全的企业间文件传输?

工业制造企业的供应商数量通常较多&#xff0c;这主要是由于工业制造行业的复杂性和多元化特点所决定的。工业制造企业的产品结构往往较为复杂&#xff0c;涉及到多种原材料、零部件和设备。这些物资的需求不仅数量大&#xff0c;而且种类繁多&#xff0c;因此需要与多个供应商…

HR招聘新员工,如何考察企业文化适配度

要解决文化适配性问题&#xff0c;那在招聘过程中一定要明确企业核心价值观。比如通过制定明确文化价值观手册的方式&#xff0c;向求职者展示企业的使命愿景和价值观。 目前最为理想的考察方式就是线上的人才测评&#xff0c;比如&#xff1a;采用职业价值观测评法&#xff0…

【HarmonyOS】头像圆形裁剪功能之手势放大缩小,平移,双击缩放控制(三)

【HarmonyOS】头像裁剪之手势放大缩小&#xff0c;平移&#xff0c;双击缩放控制&#xff08;三&#xff09; 一、DEMO效果图&#xff1a; 二、开发思路&#xff1a; 使用矩阵变换控制图片的放大缩小和平移形态。 通过监听点击手势TapGesture&#xff0c;缩放手势PinchGes…

餐厅食品留样管理系统小程序的设计

管理员账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;窗口负责人管理&#xff0c;窗口员工管理&#xff0c;冰柜管理&#xff0c;排班信息管理&#xff0c;留样食品管理&#xff0c;教育宣传管理&#xff0c;系统管理 微信端账号功能包括&#xff1a;系统…

HTML/CSS/JS学习笔记 Day2(HTML)

跟着该视频学习&#xff0c;记录笔记&#xff1a;【黑马程序员pink老师前端入门教程&#xff0c;零基础必看的h5(html5)css3移动端前端视频教程】https://www.bilibili.com/video/BV14J4114768?p12&vd_source04ee94ad3f2168d7d5252c857a2bf358 Day2 内容梳理&#xff1a;…

Python爬虫:通过js逆向获取某瓜视频的下载链接

爬虫:通过js逆向获取某瓜视频的下载链接 1. 前言2. 获取script标签下的视频加密数据3. 第一步:获取解密后的视频下载链接4. 第二步:模拟生成加密的webid值 1. 前言 就小编了解&#xff0c;某瓜视频这个网站对应视频下载链接加密处理至少经过三个版本。之前在CSDN发布了一篇关于…

【专题】2024全球电商消费电子市场研究报告合集PDF分享(附原数据表)

原文链接&#xff1a;https://tecdat.cn/?p37552 在如今数字经济蓬勃发展的大环境下&#xff0c;电商行业正以前所未有的迅猛之势&#xff0c;对全球商业版图进行着深刻的重塑。据 Stocklytics 发布的有关全球电商市场价值及未来增长趋势的专项调查报告显示&#xff0c;2024…

docker部署nginx、docker常用命令

1、安装nginx 未加版本号&#xff0c;默认最新版 docker pull nginxdocker pull nginx:版本号2、查看是否拉取成功 2-1、查看镜像 docker images2-2、镜像打包->可给他人使用 docker save -o nginx.tar nginx:latest2-3、读取打包的镜像 记得先走第三步删除镜像&#x…