1、概念:同一个线程在外层方法获取锁的时候,再进入该线程的内层方法会自动获取锁(前提是:锁对象是同一个对象),不是因为之前已经获取过还没有释放而阻塞
2、java中的ReentrantLock和synchronied都是可重入锁,可重入锁的一个优点就是可以在一定程度上避免死锁的出现
二、可重入锁的种类 一、隐式锁(Synchronized关键字使用的锁)默认是可重入锁public class ReEnterLockDemo { public synchronized void m1(){ System.out.println("外"); m2();//当获取到锁时,由于是同一个对象,所以在进入m2时可以直接进入,这就是可重入锁 } public synchronized void m2(){ System.out.println("中"); m3(); } public synchronized void m3(){ System.out.println("内"); } public static void main(String[] args) { new ReEnterLockDemo().m1(); } }
Synchronized的重入实现机制:
每个锁对象拥有一个锁计数器和一个指向持有该锁的线程的指针。
当执行monitorenter时,如果目标对象的计数器为零,你那么说明他没有被其他线程锁持有。java虚拟机会将锁对象的持有线程设置为当前线程。并且将其计数器加一。
(进入另一个带锁的方法或代码块中)在锁对象的计数器不为零的情况下,如果锁对象的持有线程是当前线程,那么java虚拟机可以将其计数器加一,否则需要等待,直至持有线程释放该锁。
当执行monitorexit时,java虚拟机则需要将锁对象计数器减一,当锁对象计数器为零时,代表锁已被释放。
二、显式锁(即Lock锁)也有ReentranLock这样的可重入锁public class ReEnterLockDemo2 { static Lock lock = new ReentrantLock(); public static void main(String[] args) { new Thread(()->{ lock.lock(); try { System.out.println("外层"); lock.lock(); try { System.out.println("中层"); }finally { lock.unlock();//加几次锁就要释放几次锁 } }finally { lock.unlock(); } }).start(); } }
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)