java线程死锁例子及解决方法

  1. Java线程死锁是由于有些资源彼此交叉取用,就有可能造成死结.  

[java] 
view plain
 copy

  1. 1线程中 取得A对象的锁定后又要取得B对象的锁定.但是同时2线程中取得B对象的锁定后又要取得A对象的锁定.这两个线程同时发生时就会造成,1线程拿到A对象锁定后等待B对象的锁定.2线程拿到B对象锁定后等待A对象锁定.这样就会进入没有停止的等待中.  

[java] 
view plain
 copy

  1. 线程死锁的一个简单例子:  

[java] 
view plain
 copy

  1. package deadLockThread;  
  2.   
  3.   
  4. public class Test {  
  5.   
  6. private int size=0;  
  7.   
  8.     public synchronized void doSome(){  
  9. size++;  
  10.     }  
  11.       
  12. public synchronized void doTest(Test test){  
  13. try {  
  14. Thread.sleep(1000);//睡眠1秒  效果更明显  
  15. catch (InterruptedException e) {  
  16. e.printStackTrace();  
  17. }  
  18.         test.doSome();   //另外一个对象调用synchronized 函数  需要取得该对象的对象锁定  
  19.   
  20.   
  21.   
  22. }  
  23.   
  24.   
  25.   
  26. }  

[java] 
view plain
 copy

  1. package deadLockThread;  
  2.   
  3. public class DeadLock {  
  4.     public static void main(String[] args) {  
  5.   
  6.         final Test t1=new Test();  
  7.         final Test t2=new Test();  
  8.           
  9.         Thread th1=new Thread(new Runnable() {  
  10.               
  11.             @Override  
  12.             public void run() {  
  13.                 // TODO Auto-generated method stub  
  14.                 t1.doTest(t2);  
  15.             }  
  16.         });  
  17.           
  18.         Thread th2=new Thread(new Runnable() {  
  19.               
  20.             @Override  
  21.             public void run() {  
  22.                 // TODO Auto-generated method stub  
  23.                 t2.doTest(t1);  
  24.             }  
  25.         });  
  26.           
  27.         th1.start();  
  28.         th2.start();  
  29.           
  30.           
  31.     }  
  32.       
  33.   
  34. }  

[java] 
view plain
 copy

  1. 如果真的没办法避免资源的交叉取用   我的解决方法是使用并行API在取得第一个对象锁定前先检测另外一个对象的对象锁定有没有被其他的线程拿走了.  

[java] 
view plain
 copy

  1. synchronized 要求线程必须取得对象锁定,才可以执行所标识的区块范围.然而使用synchronized 有许多的限制,未取得锁定的线程会直接被打断.所以我这里使用并行API Lock 替代直接操作synchronized .  

[java] 
view plain
 copy

  1. package deadLockThread;  
  2.   
  3. import java.util.concurrent.locks.Lock;  
  4. import java.util.concurrent.locks.ReentrantLock;  
  5.   
  6. public class Test {  
  7.   
  8.     private int size = 0;  
  9.   
  10.     private Lock lock = new ReentrantLock();  
  11.   
  12.     public void doSome() {  
  13.         System.out.println(Thread.currentThread().getName() + “获取size”);  
  14.         size++;  
  15.     }  
  16.   
  17.     public void doTest(Test test) {  
  18.   
  19.         while (true) {  
  20.   
  21.             boolean myLock = this.lock.tryLock();// 尝试取得当前对象的Lock锁定  
  22.             boolean testLock = test.lock.tryLock();// 尝试取得被传入得对象的Lock锁定  
  23.   
  24.             try {  
  25.   
  26.                 if (myLock && testLock) {   //当两个对象的Lock 都获取到后再进行 操作  
  27.                     test.doSome();  
  28.                     break;  
  29.                 }  
  30.   
  31.             } finally {  
  32.                 if (myLock) {  
  33.                     this.lock.unlock();  
  34.                 }  
  35.                 if (testLock) {  
  36.                     test.lock.unlock();  
  37.                 }  
  38.   
  39.             }  
  40.         }  
  41.     }  
  42.   
  43. }  

[java] 
view plain
 copy

  1. package deadLockThread;  
  2.   
  3. public class DeadLock {  
  4.     public static void main(String[] args) {  
  5.   
  6.         final Test t1 = new Test();  
  7.         final Test t2 = new Test();  
  8.   
  9.         Thread th1 = new Thread(new Runnable() {  
  10.   
  11.             @Override  
  12.             public void run() {  
  13.                 // TODO Auto-generated method stub  
  14.                 t1.doTest(t2);  
  15.             }  
  16.         });  
  17.   
  18.         Thread th2 = new Thread(new Runnable() {  
  19.   
  20.             @Override  
  21.             public void run() {  
  22.                 // TODO Auto-generated method stub  
  23.                 t2.doTest(t1);  
  24.             }  
  25.         });  
  26.   
  27.         th1.start();  
  28.         th2.start();  
  29.   
  30.     }  
  31.   
  32. }  
    原文作者:java锁
    原文地址: https://blog.csdn.net/yang5726685/article/details/78700485
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞