【java中的 嵌套管程锁死】


原文链接 
  作者:Jakob Jenkov

译者:余绍亮    校对:丁一

嵌套管程锁死类似于死锁
下面是一个嵌套管程锁死的场景 

  1. 线程1获得A对象的锁。
  2. 线程1获得对象B的锁(同时持有对象A的锁)。
  3. 线程1决定等待另一个线程的信号再继续。
  4. 线程1调用B.wait(),从而释放了B对象上的锁,但仍然持有对象A的锁。
  5. 线程2需要同时持有对象A和对象B的锁,才能向线程1发信号。
  6. 线程2无法获得对象A上的锁,因为对象A上的锁当前正被线程1持有。
  7. 线程2一直被阻塞,等待线程1释放对象A上的锁。
  8. 线程1一直阻塞,等待线程2的信号,因此,不会释放对象A上的锁,
  9. 而线程2需要对象A上的锁才能给线程1发信号……

复制代码

你可以能会说,这是个空想的场景,好吧,让我们来看看下面这个比较挫的Lock实现:

  1. //lock implementation with nested monitor lockout problem
  2. public class Lock{
  3. protected MonitorObject monitorObject = new MonitorObject();
  4. protected boolean isLocked = false;
  5. public void lock() throws InterruptedException{
  6. synchronized(this){
  7. while(isLocked){
  8. synchronized(this.monitorObject){
  9. this.monitorObject.wait();
  10. }
  11. }
  12. isLocked = true;
  13. }
  14. }
  15. public void unlock(){
  16. synchronized(this){
  17. this.isLocked = false;
  18. synchronized(this.monitorObject){
  19. this.monitorObject.notify();
  20. }
  21. }
  22. }
  23. }

复制代码

可以看到,lock()方法首先在”this”上同步,然后在monitorObject上同步。如果isLocked等于false,因为线程不会继续调用monitorObject.wait(),那么一切都没有问题 。但是如果isLocked等于true,调用lock()方法的线程会在monitorObject.wait()上阻塞。

这里的问题在于,调用monitorObject.wait()方法只释放了monitorObject上的管程对象,而与”this“关联的管程对象并没有释放。换句话说,这个刚被阻塞的线程仍然持有”this”上的锁。

(校对注:如果一个线程持有这种Lock的时候另一个线程执行了lock操作)当一个已经持有这种Lock的线程想调用unlock(),就会在unlock()方法进入synchronized(this)块时阻塞。这会一直阻塞到在lock()方法中等待的线程离开synchronized(this)块。但是,在unlock中isLocked变为false,monitorObject.notify()被执行之后,lock()中等待的线程才会离开synchronized(this)块。

简而言之,在lock方法中等待的线程需要其它线程成功调用unlock方法来退出lock方法,但是,在lock()方法离开外层同步块之前,没有线程能成功执行unlock()。

结果就是,任何调用lock方法或unlock方法的线程都会一直阻塞。这就是嵌套管程锁死。

一个更现实的例子

你可能会说,这么挫的实现方式我怎么可能会做呢?你或许不会在里层的管程对象上调用wait或notify方法,但完全有可能会在外层的this上调。

有很多类似上面例子的情况。例如,如果你准备实现一个公平锁。你可能希望每个线程在它们各自的QueueObject上调用wait(),这样就可以每次唤醒一个线程。

下面是一个比较挫的公平锁实现方式:

  1. //Fair Lock implementation with nested monitor lockout problem
  2. public class FairLock {
  3. private boolean isLocked = false;
  4. private Thread lockingThread = null;
  5. private List waitingThreads =
  6. new ArrayList();
  7. public void lock() throws InterruptedException{
  8. QueueObject queueObject = new QueueObject();
  9. synchronized(this){
  10. waitingThreads.add(queueObject);
  11. while(isLocked ||
  12. waitingThreads.get(0) != queueObject){
  13. synchronized(queueObject){
  14. try{
  15. queueObject.wait();
  16. }catch(InterruptedException e){
  17. waitingThreads.remove(queueObject);
  18. throw e;
  19. }
  20. }
  21. }
  22. waitingThreads.remove(queueObject);
  23. isLocked = true;
  24. lockingThread = Thread.currentThread();
  25. }
  26. }
  27. public synchronized void unlock(){
  28. if(this.lockingThread != Thread.currentThread()){
  29. throw new IllegalMonitorStateException(
  30. "Calling thread has not locked this lock");
  31. }
  32. isLocked = false;
  33. lockingThread = null;
  34. if(waitingThreads.size() > 0){
  35. QueueObject queueObject = waitingThread.get(0);
  36. synchronized(queueObject){
  37. queueObject.notify();
  38. }
  39. }
  40. }
  41. }

复制代码

  1. public class QueueObject {}

复制代码

乍看之下,嗯,很好,但是请注意lock方法是怎么调用queueObject.wait()的,在方法内部有两个synchronized块,一个锁定this,一个嵌在上一个synchronized块内部,它锁定的是局部变量queueObject。

当一个线程调用queueObject.wait()方法的时候,它仅仅释放的是在queueObject对象实例的锁,并没有释放”this”上面的锁。

现在我们还有一个地方需要特别注意, unlock方法被声明成了synchronized,这就相当于一个synchronized(this)块。这就意味着,如果一个线程在lock()中等待,该线程将持有与this关联的管程对象。所有调用unlock()的线程将会一直保持阻塞,等待着前面那个已经获得this锁的线程释放this锁,但这永远也发生不了,因为只有某个线程成功地给lock()中等待的线程发送了信号,this上的锁才会释放,但只有执行unlock()方法才会发送这个信号。

因此,上面的公平锁的实现会导致嵌套管程锁死。

嵌套管程锁死 VS 死锁

嵌套管程锁死与死锁很像:都是线程最后被一直阻塞着互相等待。

但是两者又不完全相同。在死锁中我们已经对死锁有了个大概的解释,死锁通常是因为两个线程获取锁的顺序不一致造成的,线程1锁住A,等待获取B,线程2已经获取了B,再等待获取A。如死锁避免中所说的,死锁可以通过总是以相同的顺序获取锁来避免。

但是发生嵌套管程锁死时锁获取的顺序是一致的。线程1获得A和B,然后释放B,等待线程2的信号。线程2需要同时获得A和B,才能向线程1发送信号。所以,一个线程在等待唤醒,另一个线程在等待想要的锁被释放。

不同点归纳如下:

  1. 死锁中,二个线程都在等待对方释放锁。
  2. 嵌套管程锁死中,线程1持有锁A,同时等待从线程2发来的信号,线程2需要锁A来发信号给线程1。

复制代码

时间: 2024-08-06 03:22:00

【java中的 嵌套管程锁死】的相关文章

死锁的三种形式:一般死锁,嵌套管程锁死,重入锁死

死锁的总结 死锁的定义: 死锁一般是指两个(或两个以上)线程同时竞争两个(或者多个)资源,从而产生同时等待的现象,使得系统僵持不动. 顺便复习一下线程与进程的定义以及他们之间的区别. 进程:一个有独立功能的程序利用某些数据资源的一次远行过程. 线程:一个进程里面的一条执行路径(或者执行过程),同一条进程下的n多条线程之间可以互相通信(共享数据). 区别:进程是相对独立的单位,进程和线程的主要差别在于它们是不同的操作系统资源管理方式. 进程有独立的地址空间,一个进程崩溃后,在保护模式下不会对其它进

Java中的嵌套类和内部类

以前看<Java编程思想>的时候,看到过嵌套类跟内部类的区别,不过后来就把它们的概念给忘了吧.昨天在看<数据结构与算法分析(Java语言版)>的时候,又遇到了这个概念,当时就很大的疑惑:嵌套类跟内部类有什么区别?只有是否有关键字static的区别吗? 所以今天找了个时间查了一下两者的详细区别,总结在这篇博客中,既方便自己的复习和学习,也启示他人吧. 1,概念: 定义在一个类内部的类,叫作"嵌套类".嵌套类分为两种:static的和非static的.后者又有一个专

java中同步嵌套引起的死锁事例代码

/* 目的:自己写一个由于同步嵌套引起的死锁! 思路:多个线程在执行时,某一时刻,0-Thread绑定了LockA锁,1-Thread绑定了LockB锁! 当0-Thread要去绑定LockB锁时 和 1-Thread要去绑定LockA锁时都不能绑定,此时两个线程不能继续进行! */ class Ticket implements Runnable{ public boolean flag; Ticket(boolean flag){ this.flag = flag; } Ticket(){

Java中的嵌套类、内部类、静态内部类

在Java中我们在一个类的内部再定义一个类,如下所示: class OuterClass { ... class NestedClass { ... } } 那么在上面的例子中我们称OuterClass为外围类(enclosing class),里面的那个类称之为嵌套类(Nested Class). 嵌套类可以分为两种,静态的和非静态的,即静态嵌套类和非静态嵌套类.非静态嵌套类又叫做内部类(Inner Class).我们通常所说的静态内部类其实是不严格的,严格的说应该叫做静态嵌套类(Static

Java中为什么要使用内部类

一.前言 关于Java的内部类,要说的东西实在太多,这篇博文中也无法一一具体说到,所以就挑些重点的讲.关于内部类的使用,你可能会疑问,为什么我们要使用内部类?为了回答这个问题,你需要知道一些关于内部类的重点.所以本篇文章首先介绍了一些关于内部类的一些与众不同的地方,后面再解答为什么我们要使用内部类这个问题.各位看官,文章稍微有点长,深吸一口气.来,我们开始吧! 二.内部类定义 内部类定义非常简单,就是把一个类的定义放在另外一个外围类定义的里面.如下面代码所示: class OutterClass

【java中重入锁死】

原文链接 作者:Jakob Jenkov 译者:刘晓日 校对:丁一 重入锁死与java中的死锁非常相似.锁和读写锁两篇文章中都有涉及到重入锁死的问题. 当一个线程重新获取锁,读写锁或其他不可重入的同步器时,就可能发生重入锁死.可重入的意思是线程可以重复获得它已经持有的锁.Java的synchronized块是可重入的.因此下面的代码是没问题的: (译者注:这里提到的锁都是指的不可重入的锁实现,并不是Java类库中的Lock与ReadWriteLock类) public class Reentra

Java 中的纤程库 – Quasar

来源:鸟窝, colobu.com/2016/07/14/Java-Fiber-Quasar/ 如有好文章投稿,请点击 → 这里了解详情 最近遇到的一个问题大概是微服务架构中经常会遇到的一个问题: 服务 A 是我们开发的系统,它的业务需要调用 B.C.D 等多个服务,这些服务是通过http的访问提供的. 问题是 B.C.D 这些服务都是第三方提供的,不能保证它们的响应时间,快的话十几毫秒,慢的话甚至1秒多,所以这些服务的Latency比较长.幸运地是这些服务都是集群部署的,容错率和并发支持都比较

类与接口(三)java中的接口与嵌套接口

一.接口 1. 接口简介 接口: 是java的一种抽象类型,是抽象方法的集合.接口比抽象类更加抽象的抽象类型. 接口语法: [修饰符] [abstract] interface 接口名 [extends 父接口名列表]{ [public] [static] [final] 常量; [public] [abstract] 方法; [public] [static] 方法{}://JDK1.8后 [public] [static] 类/接口: } 2. 接口成员的修饰符 接口中的所有成员的访问权限默

“全栈2019”Java第八十五章:实现接口中的嵌套接口

难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第八十五章:实现接口中的嵌套接口 下一章 "全栈2019"Java第八十六章:嵌套接口可以多继承吗? 学习小组 加入同步学习小组,共同交流与进步. 方式一:关注头条号Gorhaf,私信"Java学习小组". 方式二:关注公众号Gorhaf,回复"Java学习小组