浅谈可重入锁

一、可重入锁(递归锁)

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();
    }
}


版权声明:本文为weixin_42972832原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。