当某个线程请求一个由其他线程持有的锁时,发出请求的线程就会阻塞。然而,由于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并发编程实战》