[线程]线程不安全问题 --- 死锁

2024-08-30 18:12
文章标签 问题 线程 安全 死锁

本文主要是介绍[线程]线程不安全问题 --- 死锁,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 一. 引出死锁
  • 二. 可重用锁
  • 三. 死锁的三种典型场景
  • 四. 死锁产生的四个必要条件(面试题)
    • 1. 锁具有互斥特性
    • 2. 锁不可抢占(不可被剥夺)
    • 3. 请求和保持
    • 4. 循环等待
  • 五. 避免死锁问题

一. 引出死锁

class Counter{private int count;public void add(){synchronized(this){count++;}}public int get(){return count;}
}
public class Demo15 {public static void main(String[] args) throws InterruptedException {Counter counter = new Counter();Thread t1 = new Thread(() -> {for(int i = 0; i < 50000; i++){synchronized(counter){counter.add();}}});t1.start();t1.join();System.out.println("count = " + counter.get());}
}

上述代码, 我们在counter对象调用add时加了counter锁, 同时在add方法中也加counter锁, 这时我们就对同一块代码加了两层锁, 形成了锁嵌套锁的结构
在这里插入图片描述

思考:
当我们第一次对count上锁时, 是肯定会成功的
当第二次尝试加count锁时, 此时这个锁已经是被锁住的状态
按照之前的理解, 对一个已经被锁住的对象再进行加锁时, 就会出现阻塞等待
等待count锁被释放, 才能再进行加锁
但是,
要想获取到第二层锁, 就需要执行完第一层锁的大括号
要想执行完第一层锁的大括号, 就需要先获取第二层锁
这种现象就叫死锁

二. 可重用锁

运行上述代码:
在这里插入图片描述
发现并没有发生问题, 原因在于:
synchronized这个关键字, JVM在内部进行了特殊的处理
每个锁对象, 都会记录下来当前是哪个线程持有了这个锁,
当针对一个对象加锁操作时, 先会判定一下, 当前尝试加锁的线程, 是否是持有这个锁的状态,
如果没有持有这个锁, 则需要等待其他线程解锁
如果持有这个锁, 则直接放行, 就会加一遍相同的锁!!
这样的机制, 叫做==“可重用锁”==, 目的就是为了避免程序员搞出死锁

注意: 这是java锁synchronized特殊的地方, 如果是c++ / Python的锁, 嵌套锁就会发生死锁!!

三. 死锁的三种典型场景

场景一: 一个线程针对一个对象, 连续加锁(不可重入锁)两次
就是上述的问题:
如果是不可重入锁, 并且一个线程针对一个对象, 连续加锁两次, 就会引起死锁
解决方法就是引入不可重入锁

场景二: 两个线程两把锁
现在又线程1和线程2, 有两把锁A和B
两个线程先分别获取两把锁, 线程1获取A, 线程2获取B, 分别拿到锁后, 在释放之前, 再次尝试获取对方的锁

public class Demo16 {public static void main(String[] args) {Object locker1 = new Object();Object locker2 = new Object();Thread t1 = new Thread(() -> {synchronized (locker1){try {Thread.sleep(1000);//让t1等待一下t2启动, 获取locker2} catch (InterruptedException e) {throw new RuntimeException(e);}synchronized (locker2){System.out.println("t1获取了两把锁");}}});Thread t2 = new Thread(() -> {synchronized (locker2){synchronized (locker1){System.out.println("t2获取了两把锁");}}});t1.start();t2.start();}
}

此时运行发现:
在这里插入图片描述
代码正在运行, 但是什么也没打印, 说明t1t2都没拿到两把锁
t1等待t2释放locker2, t2等待t1释放locker1, 就发生了死锁

此时可以通过jconsole工具观察线程状态:
t1:
在这里插入图片描述
t2:
在这里插入图片描述
都是BLOCKED状态
场景三: N个线程, M把锁
随着线程数目 / 锁的个数增加, 此时情况就更复杂了, 就更容易出现死锁了

一个经典问题: 哲学家进餐问题
一张圆桌上面一碗面条, 假设共5个哲学家, 每个哲学家之间都有一根筷子, 哲学家可以选择:
1)思考人生(放下筷子)
2)吃面条, (此时需要拿起左右两根筷子, 才能吃面条)
在这里插入图片描述

每个哲学家啥时候吃面条, 啥时候思考人生, 这都是不确定的, 模拟线程抢占式执行
大多数情况下, 都是可以正常运行的, 只需要等待即可
但是如果出现极端情况, 就会出现问题:
如果同一时刻, 所有的哲学家都想要吃面条, 都同时拿起了左边的筷子, 那么当想要拿起右边筷子的时候, 就会发生阻塞等待, 每一个人都在等待右边的人放下筷子, 此时就会发生死锁!

四. 死锁产生的四个必要条件(面试题)

死锁, 是一个非常严重的问题!!
死锁的发生就会导致线程被卡住, 没法继续执行
同时, 死锁的产生是随机性的, 可能测试一万次都没有发生, 但是无法保证第一万零一次是否会发生死锁

死锁产生的四个必要条件:
(必要条件:缺一不可, 任何一个死锁场景,都必须具备以下四点!!!)

1. 锁具有互斥特性

这是锁的基本特点, 一个线程拿到锁后, 其他线程就得阻塞等待
(锁的基本特点)

2. 锁不可抢占(不可被剥夺)

一个线程拿到锁后, 除非自己主动释放, 别人是没法抢占的
(锁的基本特点)

3. 请求和保持

一个线程拿到一把锁后, 在不释放这个锁的前提下, 再尝试获取其他锁
(代码结构)

4. 循环等待

多个线程获取多个锁的过程中, 出现了循环等待, 如A等B, B等A
(代码结构)

五. 避免死锁问题

根据上面死锁产生的条件, 分析如何避免死锁
条件一和条件二使我们无法改变的
条件三我们可以尽量避免不让锁嵌套获取, 但是有的时候为了线程安全, 我们又必须要嵌套
条件四, 我们可以破除循环等待, 技术出现嵌套, 也不会死锁
那么我们就可以通过约定加锁顺序来避免死锁
例如上述代码:
我们约定好, 只能先加locker1, 再加locker2
在这里插入图片描述
这样就不会出现死锁了, t2只能等待t1释放locker1后, 才能继续执行在这里插入图片描述

再看哲学家就餐问题, 如果我们给每根筷子编号, 当哲学家拿筷子的时候, 只能拿编号小的筷子, 此时, 当5个人同时拿起筷子:

在这里插入图片描述
当代码中, 确实需要用到多个线程获取多把锁, 一定要约定好加锁顺序, 就可以有效避免死锁了

这篇关于[线程]线程不安全问题 --- 死锁的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1121558

相关文章

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

购买磨轮平衡机时应该注意什么问题和技巧

在购买磨轮平衡机时,您应该注意以下几个关键点: 平衡精度 平衡精度是衡量平衡机性能的核心指标,直接影响到不平衡量的检测与校准的准确性,从而决定磨轮的振动和噪声水平。高精度的平衡机能显著减少振动和噪声,提高磨削加工的精度。 转速范围 宽广的转速范围意味着平衡机能够处理更多种类的磨轮,适应不同的工作条件和规格要求。 振动监测能力 振动监测能力是评估平衡机性能的重要因素。通过传感器实时监

缓存雪崩问题

缓存雪崩是缓存中大量key失效后当高并发到来时导致大量请求到数据库,瞬间耗尽数据库资源,导致数据库无法使用。 解决方案: 1、使用锁进行控制 2、对同一类型信息的key设置不同的过期时间 3、缓存预热 1. 什么是缓存雪崩 缓存雪崩是指在短时间内,大量缓存数据同时失效,导致所有请求直接涌向数据库,瞬间增加数据库的负载压力,可能导致数据库性能下降甚至崩溃。这种情况往往发生在缓存中大量 k

客户案例:安全海外中继助力知名家电企业化解海外通邮困境

1、客户背景 广东格兰仕集团有限公司(以下简称“格兰仕”),成立于1978年,是中国家电行业的领军企业之一。作为全球最大的微波炉生产基地,格兰仕拥有多项国际领先的家电制造技术,连续多年位列中国家电出口前列。格兰仕不仅注重业务的全球拓展,更重视业务流程的高效与顺畅,以确保在国际舞台上的竞争力。 2、需求痛点 随着格兰仕全球化战略的深入实施,其海外业务快速增长,电子邮件成为了关键的沟通工具。

安全管理体系化的智慧油站开源了。

AI视频监控平台简介 AI视频监控平台是一款功能强大且简单易用的实时算法视频监控系统。它的愿景是最底层打通各大芯片厂商相互间的壁垒,省去繁琐重复的适配流程,实现芯片、算法、应用的全流程组合,从而大大减少企业级应用约95%的开发成本。用户只需在界面上进行简单的操作,就可以实现全视频的接入及布控。摄像头管理模块用于多种终端设备、智能设备的接入及管理。平台支持包括摄像头等终端感知设备接入,为整个平台提

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

2024网安周今日开幕,亚信安全亮相30城

2024年国家网络安全宣传周今天在广州拉开帷幕。今年网安周继续以“网络安全为人民,网络安全靠人民”为主题。2024年国家网络安全宣传周涵盖了1场开幕式、1场高峰论坛、5个重要活动、15场分论坛/座谈会/闭门会、6个主题日活动和网络安全“六进”活动。亚信安全出席2024年国家网络安全宣传周开幕式和主论坛,并将通过线下宣讲、创意科普、成果展示等多种形式,让广大民众看得懂、记得住安全知识,同时还

【VUE】跨域问题的概念,以及解决方法。

目录 1.跨域概念 2.解决方法 2.1 配置网络请求代理 2.2 使用@CrossOrigin 注解 2.3 通过配置文件实现跨域 2.4 添加 CorsWebFilter 来解决跨域问题 1.跨域概念 跨域问题是由于浏览器实施了同源策略,该策略要求请求的域名、协议和端口必须与提供资源的服务相同。如果不相同,则需要服务器显式地允许这种跨域请求。一般在springbo

题目1254:N皇后问题

题目1254:N皇后问题 时间限制:1 秒 内存限制:128 兆 特殊判题:否 题目描述: N皇后问题,即在N*N的方格棋盘内放置了N个皇后,使得它们不相互攻击(即任意2个皇后不允许处在同一排,同一列,也不允许处在同一斜线上。因为皇后可以直走,横走和斜走如下图)。 你的任务是,对于给定的N,求出有多少种合法的放置方法。输出N皇后问题所有不同的摆放情况个数。 输入