Java嵌套管程锁死

isen
isen
发布于 2023-09-26 / 41 阅读 / 0 评论 / 0 点赞

Java嵌套管程锁死

Nested Monitor Lockout

嵌套管程锁死

嵌套管程锁死模板

Thread 1 synchronizes on A
Thread 1 synchronizes on B (while synchronized on A)
Thread 1 decides to wait for a signal from another thread before continuing
Thread 1 calls B.wait() thereby releasing the lock on B, but not A.

Thread 2 needs to lock both A and B (in that sequence)
        to send Thread 1 the signal.
Thread 2 cannot lock A, since Thread 1 still holds the lock on A.
Thread 2 remain blocked indefinately waiting for Thread1
        to release the lock on A

Thread 1 remain blocked indefinately waiting for the signal from
        Thread 2, thereby
        never releasing the lock on A, that must be released to make
        it possible for Thread 2 to send the signal to Thread 1, etc.

实例1

//lock implementation with nested monitor lockout problem

public class Lock{
  protected MonitorObject monitorObject = new MonitorObject();
  protected boolean isLocked = false;

  public void lock() throws InterruptedException{
    synchronized(this){
      while(isLocked){
        synchronized(this.monitorObject){
            this.monitorObject.wait();
        }
      }
      isLocked = true;
    }
  }

  public void unlock(){
    synchronized(this){
      this.isLocked = false;
      synchronized(this.monitorObject){
        this.monitorObject.notify();
      }
    }
  }
}

实例2

//Fair Lock implementation with nested monitor lockout problem

public class FairLock {
  private boolean           isLocked       = false;
  private Thread            lockingThread  = null;
  private List<QueueObject> waitingThreads =
            new ArrayList<QueueObject>();

  public void lock() throws InterruptedException{
    QueueObject queueObject = new QueueObject();

    synchronized(this){
      waitingThreads.add(queueObject);

      while(isLocked || waitingThreads.get(0) != queueObject){

        synchronized(queueObject){
          try{
            queueObject.wait();
          }catch(InterruptedException e){
            waitingThreads.remove(queueObject);
            throw e;
          }
        }
      }
      waitingThreads.remove(queueObject);
      isLocked = true;
      lockingThread = Thread.currentThread();
    }
  }

  public synchronized void unlock(){
    if(this.lockingThread != Thread.currentThread()){
      throw new IllegalMonitorStateException(
        "Calling thread has not locked this lock");
    }
    isLocked      = false;
    lockingThread = null;
    if(waitingThreads.size() > 0){
      QueueObject queueObject = waitingThreads.get(0);
      synchronized(queueObject){
        queueObject.notify();
      }
    }
  }
}

public class QueueObject {}

Nested Monitor Lockout vs. Deadlock

嵌套管程锁死和死锁的结果几乎相同:所涉及的线程最终被阻塞,并且永远等待彼此。

但这两种情况并不相同。当两个线程获得锁的顺序不同时会发生死锁。 线程1锁定A,等待B,线程2已锁定B,现在等待A。并且可以通过将获取锁的顺序一致来避免死锁(锁定顺序)。 嵌套管程锁死反而是两个线程获取锁的顺序是相同的。 线程1锁定A和B,然后释放B并等待来自线程2的信号。线程2需要A和B两者向线程1发送信号。 因此,一个线程正在等待一个信号,另一个线程正在释放一个锁。

总之,在死锁中,两个线程等待对方释放锁。在嵌套管程锁死中,线程1持有锁A,并等待来自线程2的信号。线程2需要锁A来发送向线程1发出信号。