浅谈可重入锁

浅谈可重入锁,第1张

浅谈可重入锁 一、可重入锁(递归锁)

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

欢迎分享,转载请注明来源:内存溢出

原文地址: https://outofmemory.cn/zaji/5712879.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-12-17
下一篇 2022-12-17

发表评论

登录后才能评论

评论列表(0条)

保存