java 锁的可重入性

机制:每个锁都关联一个请求计数器和一个占有他的线程,当请求计数器为0时,这个锁可以被认为是unhled的,当一个线程请求一个unheld的锁时,JVM记录锁的拥有者,并把锁的请求计数加1,如果同一个线程再次请求这个锁时,请求计数器就会增加,当该线程退出syncronized块时,计数器减1,当计数器为0时,锁被释放。

         java锁的可重入性机制可以解决下面这个问题

[java]  view plain copy
 
  1. public class Widget {  
  2.     public synchronized void doSomething() {  
  3.         ...  
  4.     }  
  5. }  
  6.   
  7. public class LoggingWidget extends Widget {  
  8.     public synchronized void doSomething() {  
  9.         System.out.println(toString() + ": calling doSomething");  
  10.         super.doSomething();  
  11.     }  
  12. }  

如果没有Java锁的可重入性,当一个线程获取LoggingWidget的doSomething()代码块的锁后,这个线程已经拿到了LoggingWidget的锁,当调用父类中的doSomething()方法的时,JVM会认为这个线程已经获取了LoggingWidget的锁,而不能再次获取,从而无法调用Widget的doSomething()方法,从而照成死锁。从中我们也能看出,java线程是基于“每线程(per-thread)”,而不是基于“每调用的(per-invocation)”的,也就是说java为每个线程分配一个锁,而不是为每次调用分配一个锁。

你可能感兴趣的:(thread)