当前位置: 首页 > news >正文

Java多线程编程中的线程死锁

Java多线程编程中的线程死锁

在多线程编程中,线程死锁是一种常见的问题,它发生在两个或多个线程互相等待对方释放资源的情况下,导致程序无法继续执行。本文将介绍线程死锁的概念、产生原因、示例以及如何预防和解决线程死锁问题。


线程死锁的概念

线程死锁是指两个或多个线程被阻塞,它们互相等待对方释放所持有的资源,导致程序无法继续执行。通常,死锁发生在多个线程试图获取一组共享资源时,这些资源已被其他线程锁定,而这些线程又在等待其他线程释放资源。


线程死锁的产生原因

线程死锁通常由以下四个条件共同导致:

  1. 互斥条件: 至少有一个资源被限定为一次只能被一个线程持有。
  2. 请求与保持条件: 一个线程持有至少一个资源并请求其他线程持有的资源。
  3. 不可剥夺条件: 已经获得的资源在没有被释放之前,不能被其他线程剥夺。
  4. 循环等待条件: 多个线程形成一种循环等待资源的关系。

线程死锁的示例

以下是一个简单的线程死锁示例:

public class DeadlockDemo {public static void main(String[] args) {final Object resource1 = "resource1";final Object resource2 = "resource2";Thread thread1 = new Thread(() -> {synchronized (resource1) {System.out.println("Thread 1: Holding resource 1...");try { Thread.sleep(100); } catch (InterruptedException e) {}System.out.println("Thread 1: Waiting for resource 2...");synchronized (resource2) {System.out.println("Thread 1: Holding resource 1 and 2...");}}});Thread thread2 = new Thread(() -> {synchronized (resource2) {System.out.println("Thread 2: Holding resource 2...");try { Thread.sleep(100); } catch (InterruptedException e) {}System.out.println("Thread 2: Waiting for resource 1...");synchronized (resource1) {System.out.println("Thread 2: Holding resource 1 and 2...");}}});thread1.start();thread2.start();}
}

输出结果如下:因为俩个同步块之间都嵌套其他的锁,因此先入死循环,同步块没结束,资源锁没办法被释放。

在这里插入图片描述


预防和解决线程死锁

要预防和解决线程死锁问题,可以采取以下几种方法:
  1. 避免循环等待: 尽量按照相同的顺序获取资源,减少死锁的可能性。
  2. 使用定时锁: 在获取锁时,添加超时机制,避免永久等待。
  3. 使用资源分级: 将资源按优先级进行划分,先获取低级别资源再获取高级别资源。
  4. 使用工具: 使用工具分析和检测潜在的死锁问题。

当涉及到线程死锁时,还有一个典型的例子是“哲学家就餐问题”,这个问题可以用来说明线程死锁的发生。

​ 在这个问题中,有五位哲学家围坐在一个圆桌旁边,每位哲学家面前有一盘意大利面和一只叉子。哲学家们交替思考和进食,思考时不需要叉子,进食时需要用两只叉子。然而,只有五只叉子可供使用。问题的关键在于,当每位哲学家都持有一只叉子并等待另一只叉子时,就可能发生死锁。

下面是一个简化的示例代码,演示了哲学家就餐问题导致的线程死锁:
public class DiningPhilosophersDeadlock {public static class Philosopher extends Thread {private Object leftFork;private Object rightFork;public Philosopher(Object leftFork, Object rightFork) {this.leftFork = leftFork;this.rightFork = rightFork;}public void run() {synchronized (leftFork) {System.out.println(Thread.currentThread().getName() + " 拿起左叉子");try {Thread.sleep(100); // 模拟思考时间} catch (InterruptedException e) {e.printStackTrace();}synchronized (rightFork) {System.out.println(Thread.currentThread().getName() + " 拿起右叉子,开始进食");}}}}public static void main(String[] args) {int numPhilosophers = 5;Philosopher[] philosophers = new Philosopher[numPhilosophers];Object[] forks = new Object[numPhilosophers];for (int i = 0; i < numPhilosophers; i++) {forks[i] = new Object();}for (int i = 0; i < numPhilosophers; i++) {Object leftFork = forks[i];Object rightFork = forks[(i + 1) % numPhilosophers];philosophers[i] = new Philosopher(leftFork, rightFork);philosophers[i].start();}}
}

在这个例子中,五位哲学家(线程)围坐在圆桌上,每位哲学家需要持有其左边和右边的叉子才能进食。当每位哲学家都持有一只叉子并等待另一只叉子时,就会出现死锁。

输出结果可能类似于(顺序可能会有所不同):

Thread-0 拿起左叉子
Thread-1 拿起左叉子
Thread-2 拿起左叉子
Thread-3 拿起左叉子
Thread-4 拿起左叉子

在这个阶段,每位哲学家都持有左边的叉子,但都在等待右边的叉子,导致了线程死锁。

这个例子展示了多线程中常见的死锁情况,其中每位哲学家代表一个线程,而叉子则代表共享资源。要解决这个问题,可以使用各种方法,如调整锁的获取顺序、引入超时机制、或者使用更高级的同步机制来避免死锁的发生。


总结

PS:线程死锁是多线程编程中的一个常见问题,它发生在多个线程互相等待对方释放资源的情况下,导致程序无法继续执行。了解线程死锁的产生原因和示例,以及预防和解决线程死锁的方法,有助于帮助我们编写更多更加优良的多线程程序。

作者:Stevedash

发表于:2023年8月14日 20点25分

来源:Java 多线程编程 | 菜鸟教程 (runoob.com)

注:本文内容基于个人学习理解,如有错误或疏漏,欢迎指正。感谢阅读!如果觉得有帮助,请点赞和分享。

http://www.lryc.cn/news/123617.html

相关文章:

  • 在浏览器中使用javascript打印HTML中指定Div带背景图片内容生成PDF电子证书查询的解决方案
  • 【Redis实践篇】使用Redisson 优雅实现项目实践过程中的5种场景
  • 污水处理厂人员定位方案介绍
  • 约数个数(质因子分解)
  • 【QT】 QSS样式表设计一文了解
  • 9-AJAX-1入门
  • ssh免密登录
  • 全球公链周进展-2023/8/14
  • python装饰器详解,python装饰器笔记心得
  • 【C语言】每日一题(寻找数组的中心下标)
  • centos 安装 nginx配置ssl 和 获取用户真实ip
  • RocketMQ 消息消费 轮询机制 PullRequestHoldService
  • springboot 数据库版本升级管理常用解决方案
  • 78. 子集
  • Mask RCNN网络结构以及整体流程的详细解读
  • Android Framework底层原理之WMS的启动流程
  • Leaflet入门,Leaflet加载xyz地图,以vue-leaflet插件加载高德地图为例
  • 【ARM Cache 系列文章 8 -- ARM DynamIQ 技术介绍
  • 24届近5年南京大学自动化考研院校分析
  • 微信小程序(原生)和uniapp预览电子文件doc/pdf/ppt/excel等
  • 【前端 | CSS】align-items与align-content的区别
  • Go语言入门
  • Python学习笔记第五十五天(Pandas CSV文件)
  • 自然语言处理: 第七章GPT的搭建
  • 【奶奶看了都会】2分钟学会制作最近特火的ikun幻术图
  • 【深度学习】【风格迁移】Zero-shot Image-to-Image Translation
  • Day 30 C++ STL 常用算法(上)
  • MES系统在机器人行业生产管理种的运用
  • Spark(39):Streaming DataFrame 和 Streaming DataSet 输出
  • 【云原生】Docker 详解(一):从虚拟机到容器