返回顶部
首页 > 资讯 > 精选 >怎么避免java死锁
  • 357
分享到

怎么避免java死锁

2023-06-16 18:06:39 357人浏览 独家记忆
摘要

这篇文章主要介绍“怎么避免java死锁”,在日常操作中,相信很多人在怎么避免java死锁问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么避免java死锁”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!来到

这篇文章主要介绍“怎么避免java死”,在日常操作中,相信很多人在怎么避免java死锁问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么避免java死锁”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

来到钱庄,告诉柜员你要给铁蛋儿转 100 铜钱,这时柜员转身在墙上寻找你和铁蛋儿的账本,此时柜员可能面临三种情况:

  1.  理想状态: 你和铁蛋儿的账本都是空闲状态,一起拿回来,在你的账本上减 100 铜钱,在铁蛋儿账本上加 100 铜钱,柜员转身将账本挂回到墙上,完成你的业务

  2.  尴尬状态: 你的账本在,铁蛋儿的账本被其他柜员拿出去给别人转账,你要等待其他柜员把铁蛋儿的账本归还

  3.  抓狂状态: 你的账本不在,铁蛋儿的账本也不在,你只能等待两个账本都归还

放慢柜员的取账本操作,他一定是先拿到你的账本,然后再去拿铁蛋儿的账本,两个账本都拿到(理想状态)之后才能完成转账,用程序模型来描述一下这个拿取账本的过程:

怎么避免java死锁

我们继续用程序代码描述一下上面这个模型:

class Account {    private int balance;    // 转账    void transfer(Account target, int amt){      // 锁定转出账户      synchronized(this) {                      // 锁定转入账户        synchronized(target) {            if (this.balance > amt) {            this.balance -= amt;            target.balance += amt;          }        }      }    }   }

这个解决方案看起来很不错,解决了文章开头说的两个问题,但真是这样吗?

我们刚刚说过的理想状态是钱庄只有一个柜员(既单线程)。随着钱庄规模变大,墙上早已挂了非常多个账本,钱庄为了应对繁忙的业务,开通了多个窗口,此时有多个柜员(多线程)处理钱庄业务。

柜员 1 正在办理给铁蛋儿转账的业务,但只拿到了你的账本;柜员 2 正在办理铁蛋儿给你转账的业务,但只拿到了铁蛋儿的账本,此时双方出现了尴尬状态,两位柜员都在等待对方归还账本为当前客户办理转账业务。

怎么避免java死锁

现实中柜员会沟通,喊出一嗓子 老铁,铁蛋儿的账本先给我用一下,用完还给你,但程序却没这么智能,synchronized 内置锁非常执着,它会告诉你「死等」的道理,最终出现死锁

Java 有了 synchronized 内置锁,还发明了显示锁 Lock,是不是就为了治一治 synchronized 「死等」的执着呢??

解决方案

如何解决上面的问题呢?正所谓知己知彼方能百战不殆,我们要先了解什么情况会发生死锁,才能知道如何避免死锁,很幸运我们可以站在巨人的肩膀上看待问题

Coffman 总结出了四个条件说明可以发生死锁的情形:

Coffman 条件

互斥条件:指进程对所分配到的资源进行排它性使用,即在一段时间内某资源只由一个进程占用。如果此时还有其它进程请求资源,则请求者只能等待,直至占有资源的进程用毕释放。

请求和保持条件:指进程已经保持至少一个资源,但又提出了新的资源请求,而该资源已被其它进程占有,此时请求进程阻塞,但又对自己已获得的其它资源保持不放。

不可剥夺条件:指进程已获得的资源,在未使用完之前,不能被剥夺,只能在使用完时由自己释放。

环路等待条件:指在发生死锁时,必然存在一个进程——资源的环形链,即进程集合{P1,P2,···,Pn}中的 P1 正在等待一个 P2 占用的资源;P2 正在等待 P3 占用的资源,……,Pn 正在等待已被 P0 占用的资源。

怎么避免java死锁

这几个条件很好理解,其中「互斥条件」是并发编程的根基,这个条件没办法改变。但其他三个条件都有改变的可能,也就是说破坏另外三个条件就不会出现上面说到的死锁问题

破坏请求和保持条件

每个柜员都可以取放账本,很容易出现互相等待的情况。要想破坏请求和保持条件,就要一次性拿到所有资源。

作为程序猿你一定听过这句话:

任何软件工程遇到的问题都可以通过增加一个中间层来解决

我们不允许柜员都可以取放账本,账本要由单独的账本管理员来管理

怎么避免java死锁

也就是说账本管理员拿取账本是临界区,如果只拿到其中之一的账本,那么不会给柜员,而是等待柜员下一次询问是否两个账本都在

//账本管理员  public class AccountBookManager {      synchronized boolean getAllRequiredAccountBook( Object from, Object to){          if(拿到所有账本){              return true;          } else{              return false;          }      }      // 归还资源      synchronized void releaseObtainedAccountBook(Object from, Object to){          归还获取到的账本      }  }  public class Account {      //单例的账本管理员      private AccountBookManager accountBookManager;      public void transfer(Account target, int amt){          // 一次性申请转出账户和转入账户,直到成功          while(!accountBookManager.getAllRequiredAccountBook(this, target)){              return;          }          try{              // 锁定转出账户              synchronized(this){                  // 锁定转入账户                  synchronized(target){                      if (this.balance > amt){                          this.balance -= amt;                          target.balance += amt;                      }                  }              }          } finally {              accountBookManager.releaseObtainedAccountBook(this, target);          }      }  }

破坏不可剥夺条件

上面已经给了你小小的提示,为了解决内置锁的执着,Java 显示锁支持通知(notify/notifyall)和等待(wait),也就是说该功能可以实现喊一嗓子 老铁,铁蛋儿的账本先给我用一下,用完还给你 的功能,这个后续将到 Java SDK 相关内容时会做说明

破坏环路等待条件

破坏环路等待条件也很简单,我们只需要将资源序号大小排序获取就会解决这个问题,将环路拆除

怎么避免java死锁

继续用代码来说明:

class Account {    private int id;    private int balance;    // 转账    void transfer(Account target, int amt){      Account smaller = this              Account larger = target;          // 排序      if (this.id > target.id) {         smaller = target;                   larger = this;                  }                                // 锁定序号小的账户      synchronized(smaller){        // 锁定序号大的账户        synchronized(larger){           if (this.balance > amt){            this.balance -= amt;            target.balance += amt;          }        }      }    }   }

当 smaller 被占用时,其他线程就会被阻塞,也就不会存在死锁了.

附加说明

在实际业务中,关于 Account 都会是数据库对象,我们可以通过事务数据库的乐观锁来解决的。另外分布式系统中,账本管理员这个角色的处理也可能会用 redis 分布式锁来解决.

在处理破坏请求和保持条件时,我们使用的是 while 循环方式来不断请求锁的时候,在实际业务中,我们会有 timeout 的设置,防止无休止的浪费 CPU 使用率

另外大家可以尝试使用阿里开源工具 Arthas 来查看 CPU 使用率,线程等相关问题,GitHub 上有明确的说明

public void transfer(Account target, int amt){      // 一次性申请转出账户和转入账户,直到成功      while(accountBookManager.getAllRequiredAccountBook(this, target)){}          try{              // 锁定转出账户              synchronized(this){                  // 锁定转入账户                  synchronized(target){                      if (this.balance > amt){                          this.balance -= amt;                          target.balance += amt;                      }                  }              }          } finally {              accountBookManager.releaseObtainedAccountBook(this, target);          }      }  }

到此,关于“怎么避免java死锁”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

--结束END--

本文标题: 怎么避免java死锁

本文链接: https://lsjlt.com/news/284772.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 怎么避免java死锁
    这篇文章主要介绍“怎么避免java死锁”,在日常操作中,相信很多人在怎么避免java死锁问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么避免java死锁”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!来到...
    99+
    2023-06-16
  • 怎么在Java中避免死锁
    这篇文章给大家介绍怎么在Java中避免死锁,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Java的特点有哪些Java的特点有哪些1.Java语言作为静态面向对象编程语言的代表,实现了面向对象理论,允许程序员以优雅的思维...
    99+
    2023-06-14
  • java中怎么避免程序死锁
    本篇文章为大家展示了java中怎么避免程序死锁,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。  死锁发生的条件  互斥,共享资源只能被一个线程占用  占有且等待,线程 t1 已经取得共享资源 s1,...
    99+
    2023-06-02
  • Golang怎么实现死锁?怎么避免?
    Golang 是一门高效、强类型、静态类型的编程语言,由 Google 开发,目的是为了解决一些类似于死锁等问题。虽然 Golang 实现死锁十分困难,但是在这篇文章中,我们将要探讨如何使用 Golang 实现死锁。什么是死锁?死锁指的是多...
    99+
    2023-05-14
  • MySQL死锁怎么检测和避免
    这篇文章主要讲解了“MySQL死锁怎么检测和避免”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“MySQL死锁怎么检测和避免”吧! ...
    99+
    2024-04-02
  • mysql怎么避免死锁的产生
    这篇文章主要讲解了“mysql怎么避免死锁的产生”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“mysql怎么避免死锁的产生”吧!说明等待事务超时,主动回滚。进行死锁检查,主动回滚某一事务,让...
    99+
    2023-06-20
  • 如何避免mysql死锁
    这篇文章主要介绍了如何避免mysql死锁,具有一定借鉴价值,需要的朋友可以参考下。希望大家阅读完这篇文章后大有收获。下面让小编带着大家一起了解一下。如何尽可能避免死锁:1、以固定的顺序访问表和行。比如两个更...
    99+
    2024-04-02
  • java并发编程死锁定义及如何避免死锁
    目录场景模拟分析场景一:狭路相逢场景二:冷战场景三:哲学家就餐场景四:竞争资源死锁是什么?产生死锁的的四个条件如下:如何避免死锁?方案一:破坏不剥夺条件方案二:破坏请求与保持条件方案...
    99+
    2024-04-02
  • golang中什么是死锁?怎么避免?
    一、什么是死锁?在并发编程中,当两个或多个进程(线程)相互等待对方的资源释放时,就会产生死锁(Deadlock)。简单来说,就是每个进程都占用了一部分资源,同时需要等待对方释放资源,以完成自己的任务,但是对方也一样,这时候就会造成死循环式的...
    99+
    2023-05-14
  • java并发编程死锁定义及避免死锁案例分析
    这篇文章主要介绍“java并发编程死锁定义及避免死锁案例分析”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“java并发编程死锁定义及避免死锁案例分析”文章能帮助大家解决问题。场景模拟分析场景一:狭路...
    99+
    2023-06-29
  • MySQL的死锁机制以及避免死锁的方法
    本篇内容主要讲解“MySQL的死锁机制以及避免死锁的方法”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“MySQL的死锁机制以及避免死锁的方法”吧! ...
    99+
    2024-04-02
  • Java多线程死锁如何避免及解决
    本篇内容介绍了“Java多线程死锁如何避免及解决”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!下面是一个多线程死锁的例子输出thread1&...
    99+
    2023-06-02
  • Java 同步问题:如何避免 IDE 中的死锁?
    在 Java 开发中,同步问题是一个非常常见的问题。其中,死锁是最棘手的问题之一。在 IDE 中,由于代码的复杂性和多线程的并发性,死锁问题更加常见。本文将介绍 Java 中的同步问题,包括死锁问题,以及如何避免 IDE 中的死锁。 一、...
    99+
    2023-06-26
    同步 ide http
  • Java中避免出现死锁的方法有哪些
    今天就跟大家聊聊有关Java中避免出现死锁的方法有哪些,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。避免死锁的技术:加锁顺序加锁时限死锁检测加锁顺序当多个线程需要相同的一些锁,但是按...
    99+
    2023-05-31
    java 死锁 ava
  • Go 生产者消费者避免死锁
    问题内容 我有一个关于消费者和生产者的代码。虽然我在这里提出了这个问题以进行代码审查,并且这个想法的很大一部分是从这个线程中衍生出来的,这里是操场上的代码。 此代码有多个生产者和消费...
    99+
    2024-02-06
  • mysql怎么避免锁表
    mysql避免锁表的方法:保持事务的轻量,可减少对锁资源占用。尽量避免并发的执行涉及到修改数据的语句。避免使用子查询,尽量使用主键。编写应用程序过程中,让进程持有锁的时间缩短。...
    99+
    2024-04-02
  • 分时操作系统如何避免死锁?
    死锁概述 死锁是一个计算机系统中的常见问题,它会导致系统无法再继续执行。死锁发生在多个进程同时请求有限的资源时,导致它们都不能获得所需的资源并无限等待。在这种情况下,系统将陷入死锁状态,无法继续进行任何操作。 分时操作系统中的死锁避免机...
    99+
    2024-03-05
    死锁、分时操作系统、资源管理、预防、避免、检测、恢复
  • java多线程学习之死锁的模拟和避免(实例讲解)
    1.死锁死锁是这样一种情形:多个线程同时被阻塞,它们中的一个或者全部都在等待某个资源被释放。由于线程被无限期地阻塞,因此程序不可能正常终止。Java 死锁产生的四个必要条件:互斥使用,即当资源被一个线程使用(占有)时,别的线程不能使用不可抢...
    99+
    2023-05-31
    java 多线程 避免
  • MySQL死锁使用详解及检测和避免方法
    目录前言锁的释放与阻塞死锁的发生和检测查看锁信息(日志)死锁的避免前言 上一篇博客我们知道的mysql事务的隔离机制和实现,以及锁的详细解析 链接: MySQL脏读幻读不可重复读及事务的隔离级别和MVCC、LBCC实现 ...
    99+
    2022-07-11
    MySQL死锁使用 MySQL死锁检测 MySQL死锁避免方法
  • 死锁(JAVA)
    死锁在多线程代码中是非常严重的BUG,一旦代码中出现死锁就会导致线程卡死。 当单个线程连续两次对同一个对象进行加锁操作时,如果该锁是不可重入锁就会发生死锁(线程卡死) 两个线程两把锁,如果出现这种情况也是会发生死锁:线程t1已经获取了锁A,...
    99+
    2023-12-22
    java java-ee
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作