加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 站长学院 > PHP教程 > 正文

【Java并发编程】之九:死锁(含代码)

发布时间:2020-12-13 20:06:13 所属栏目:PHP教程 来源:网络整理
导读:当线程需要同时持有多个锁时,有可能产生死锁。斟酌以下情形: 线程A当前持有互斥所锁lock1,线程B当前持有互斥锁lock2。接下来,当线程A依然持有lock1时,它试图获得lock2,由于线程B正持有lock2,因此线程A会阻塞等待线程B对lock2的释放。如果此时线程B在

当线程需要同时持有多个锁时,有可能产生死锁。斟酌以下情形:

      线程A当前持有互斥所锁lock1,线程B当前持有互斥锁lock2。接下来,当线程A依然持有lock1时,它试图获得lock2,由于线程B正持有lock2,因此线程A会阻塞等待线程B对lock2的释放。如果此时线程B在持有lock2的时候,也在试图获得lock1,由于线程A正持有lock1,因此线程B会阻塞等待A对lock1的释放。2者都在等待对方所持有锁的释放,而2者却又都没释放自己所持有的锁,这时候2者便会1直阻塞下去。这类情形称为死锁。

      下面给出1个两个线程间产生死锁的示例,以下:

[java] view plaincopy

在CODE上查看代码片

派生到我的代码片

  1. public class Deadlock extends Object {  
  2.     private String objID;  
  3.   
  4. public Deadlock(String id) {  
  5.         objID = id;  
  6.     }  
  7. synchronized void checkOther(Deadlock other) {  
  8.         print("entering checkOther()");  
  9.         try { Thread.sleep(2000); }   
  10.         catch ( InterruptedException x ) { }  
  11.         print("in checkOther() - about to " + "invoke 'other.action()'");  
  12.         //调用other对象的action方法,由于该方法是同步方法,因此会试图获得other对象的对象锁  
  13.         other.action();  
  14.         print("leaving checkOther()");  
  15.     }  
  16.   
  17.     void action() {  
  18.         print("entering action()");  
  19. 500); }   
  20. catch ( InterruptedException x ) { }  
  21.         print("leaving action()");  
  22. void print(String msg) {  
  23.         threadPrint("objID=" + objID + " - " + msg);  
  24. static void threadPrint(String msg) {  
  25.         String threadName = Thread.currentThread().getName();  
  26.         System.out.println(threadName + ": " + msg);  
  27. void main(String[] args) {  
  28. final Deadlock obj1 = new Deadlock("obj1");  
  29. final Deadlock obj2 = new Deadlock("obj2");  
  30.         Runnable runA = new Runnable() {  
  31.                 void run() {  
  32.                     obj1.checkOther(obj2);  
  33.                 }  
  34.             };  
  35.         Thread threadA = new Thread(runA, "threadA");  
  36.         threadA.start();  
  37. 200); }   
  38.         Runnable runB =                      obj2.checkOther(obj1);  
  39.         Thread threadB = new Thread(runB, "threadB");  
  40.         threadB.start();  
  41. 5000); }   
  42.         threadPrint("finished sleeping");  
  43.         threadPrint("about to interrupt() threadA");  
  44.         threadA.interrupt();  
  45. 1000); }   
  46.         threadPrint("about to interrupt() threadB");  
  47.         threadB.interrupt();  
  48.         threadPrint("did that break the deadlock?");  
  49. }  

     运行结果以下:


     从结果中可以看出,在履行到other.action()时,由于两个线程都在试图获得对方的锁,但对方都没有释放自己的锁,因此便产生了死锁,在主线程中试图中断两个线程,但都无果。


     大部份代码其实不容易产生死锁,死锁可能在代码中隐藏相当长的时间,等待不常见的条件地产生,但即便是很小的几率,1旦产生,即可能造成毁灭性的破坏。避免死锁是1件困难的事,遵守以下原则有助于规避死锁: 

     1、只在必要的最短时间内持有锁,斟酌使用同步语句块代替全部同步方法;

     2、尽可能编写不在同1时刻需要持有多个锁的代码,如果不可避免,则确保线程持有第2个锁的时间尽可能短暂;

     3、创建和使用1个大锁来代替若干小锁,并把这个锁用于互斥,而不是用作单个对象的对象级别锁;

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读