内部锁synchronized ----重进入(Reentrancy)

一、可重进入的概念 ( Reentrancy )


      当一个线程请求其他线程已经占有的锁时,请求线程会被阻塞。然而 java 的内部锁 synchronized 是可重进入的,因此线程在试图获得它自己占有的锁时,请求会成功。

      重进入是指请求是基于线程的,而不是基于调用的。重进入的实现是通过每个锁关联一个请求计数和一个占有它的线程。当计数为0时,认为锁是未被占用的。线程请求一个未被占用的锁时, JVM 将记录锁的占有者,并将请求计数置为1。如果同一线程再次请求这个锁,计数将递增;每次占用线程退出同步块,计数器值将递减。直到计数器达到0时,锁被释放。


二、可重进入的意义


      重进入方便了锁行为的封装,因此简化了面向对象并发代码的开发。例如下面的代码,子类覆写了父类  synchronized 类型的方法,并调用了父类中的方法。如果没有可重入的锁,这段看上去没问题的代码就会产生死锁。因为 Widget 和 LoggingWidget 中的 doSomething 方法都是 synchronized 类型的,都会在处理前试图获得 Widget 的锁。如果内部锁不是可重入的, super.doSomething 的调用者将永远无法得到 Widget 的锁,因为锁已经被占用,导致线程会永久地延迟,等待着一个永远无法获得的锁。重进入帮助我们避免了这种死锁。

public class Widget{
    public synchronized void doSomething(){
        ...
    }
}

public class LoggingWidget extends Widget{
    public synchronized void doSomething(){
        System.out.println(toString() + ":calling doSomething");
        super.doSomething();
    }
}


你可能感兴趣的:(java并发)