java内置锁synchronized的可重入性

当某个线程请求一个由其他线程持有的锁时,发出请求的线程就会阻塞。然而,由于java内置锁synchronized是可重入的,因此如果某个线程试图获得一个已经有它自己持有的锁,那么这个请求就会成功。如代码:

public class Widget {
    public synchronized void doSomething() {
        ...
    }
}
 
public class LoggingWidget extends Widget {
    public synchronized void doSomething() {
        System.out.println(toString() + ": calling doSomething");
        super.doSomething();//若内置锁是不可重入的,则发生死锁
    }
}

1.重入

“重入”意味着获取锁操作的粒度是“线程”,而不是“调用”。

2.重入的实现方式

为每个锁关联一个获取计数值和一个所有者线程。当计数值为0的时,就认为这个锁是没有被任何线程持有。当线程请求一个未被持有的锁时,对于内置锁synchronized,jvm将记下锁的持有者,并且将获取计数值置为1。如果同一个线程再次获取这个锁,计数值将递增,而当线程线程推出同步代码块时,计数值将递减。当计数值为0时,这个锁将被释放。

3.java的synchronized关键字

java内置锁synchronized的可重入性有jvm保证的,这个提升了加锁行为的封装性,从而简化了java并发编程的开发。但Tiger后提供了可重入锁ReentrantLock是自己实现可重入性的,而不是由jvm保证的。关于ReentrantLock会在后面介绍的。

4.java内置锁synchronized的可重入性可以降低deadLock的情况发生,如上面的事例代码。


参考书籍:《java并发编程实战》

你可能感兴趣的:(书籍笔记)