死锁
来源:互联网 发布:阿里云服务器购买 编辑:程序博客网 时间:2024/06/06 15:02
当一个线程永远地持有一个锁,并且其他线程都尝试去获得这个锁时,那么它们将永远被阻塞,这个我们都知道。如果线程A持有锁L并且想获得锁M,线程B持有锁M并且想获得锁L,那么这两个线程将永远等待下去,这种情况就是最简单的死锁形式。
在数据库系统的设计中考虑了监测死锁以及从死锁中恢复,数据库如果监测到了一组事物发生了死锁时,将选择一个牺牲者并放弃这个事物。Java虚拟机解决死锁问题方面并没有数据库这么强大,当一组Java线程发生死锁时,这两个线程就永远不能再使用了,并且由于两个线程分别持有了两个锁,那么这两段同步代码/代码块也无法再运行了—-除非终止并重启应用。
死锁是设计的BUG,问题比较隐晦。不过死锁造成的影响很少会立即显现出来,一个类可能发生死锁,并不意味着每次都会发生死锁,这只是表示有可能。当死锁出现时,往往是在最糟糕的情况—-高负载的情况下。
下面给出一个产生死锁的简单代码并且演示如何分析这是一个死锁:
package com.mook.deadLock;public class DeadLock{ private final Object left = new Object(); private final Object right = new Object(); public void leftRight() throws Exception { synchronized (left) { Thread.sleep(2000); synchronized (right) { System.out.println("leftRight end!"); } } } public void rightLeft() throws Exception { synchronized (right) { Thread.sleep(2000); synchronized (left) { System.out.println("rightLeft end!"); } } }}
package com.mook.deadLock;public class Thread0 extends Thread{ private DeadLock dl; public Thread0(DeadLock dl) { this.dl = dl; } public void run() { try { dl.leftRight(); } catch (Exception e) { e.printStackTrace(); } }}
package com.mook.deadLock;public class Thread1 extends Thread{ private DeadLock dl; public Thread1(DeadLock dl) { this.dl = dl; } public void run() { try { dl.rightLeft(); } catch (Exception e) { e.printStackTrace(); } }}
package com.mook.deadLock;public class Test { public static void main(String[] args) { DeadLock dl = new DeadLock(); Thread0 t0 = new Thread0(dl); Thread1 t1 = new Thread1(dl); t0.start(); t1.start(); while(true); }}
至于结果,没有结果,什么语句都不会打印,因为死锁了。下面演示一下如何定位死锁问题:
1、jps获得当前Java虚拟机进程的pid
jps(Java Virtual Machine Process Status Tool)是JDK 1.5提供的一个显示当前所有Java进程pid的命令
其中AppMain就是该程序运行的java进程
2、jstack打印堆栈。
jstack打印内容的最后其实已经报告发现了一个死锁
避免死锁的方式
既然可能产生死锁,那么接下来,讲一下如何避免死锁。
1、让程序每次至多只能获得一个锁。当然,在多线程环境下,这种情况通常并不现实
2、设计时考虑清楚锁的顺序,尽量减少嵌在的加锁交互数量
3、既然死锁的产生是两个线程无限等待对方持有的锁,那么只要等待时间有个上限不就好了。当然synchronized不具备这个功能,但是我们可以使用Lock类中的tryLock方法去尝试获取锁,这个方法可以指定一个超时时限,在等待超过该时限之后变回返回一个失败信息
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 死锁
- 解决数据库连接报错Listener refused the connection with the following error: ORA-12505
- Hydra使用
- sql
- 【51单片机学习笔记】彩色呼吸灯
- Zookeeper集群及单机安装
- 死锁
- java for的几种循环方式
- C#DES文件加解密工具类
- [模板]treap小结
- String StringBuffer 和StringBuild之间的区别
- 我的第一个SpringBoot应用
- Angular 元数据
- 设计模式——单件模式(Singleton Pattern)
- Java中String、StringBuffer、StringBuilder的区别