面试:什么是可重入性?为什么 synchronized 是可重入锁?

一、什么是可重入性

可重入锁,就是同一个线程在外层方法获取锁之后,再次进入该锁的内层同步代码时,可以直接再次获取锁,不会被自己阻塞。
简单说:自己加的锁,自己可以重复进,不会死锁自己。

二、为什么说 synchronized 是可重入锁

因为 synchronized 底层是基于 Monitor 监视器 + 锁计数器 实现的:

线程第一次获取锁时,计数器置为 1

同一线程再次进入同步代码时,计数器 +1

每退出一层同步代码,计数器 -1

计数器为 0 时,才真正释放锁

所以同一个线程可以重复获取同一把锁,不会产生死锁,因此 synchronized 是可重入锁。

用代码举个例子就是:

复制代码
public class Test {
    public synchronized void a() {
        b(); // 同一线程,再次进入 synchronized
    }

    public synchronized void b() {
        // ...
    }
}

调用 a() 时线程已经获取锁,再调用 b() 不会阻塞,这就是可重入。

顺便记一个加分点:
ReentrantLock 也是可重入锁,原理一样,靠 AQS 里的 state 计数器实现。
不可重入锁会导致:自己锁自己 → 死锁。

而对于这种不可重入锁最通用的解决办法就是使用可重入锁,通过记录锁持有线程和计数器,避免同一线程重复加锁时被阻塞。

相关推荐
葫芦和十三5 小时前
图解 MongoDB 18|复制集拓扑:Primary、Secondary 和 Arbiter 的分工
后端·mongodb·面试
狼爷11 小时前
吃透 Java Function 接口,搞定 99% 的 Stream 场景
java·函数式编程
葫芦和十三11 小时前
图解 MongoDB 15|journal 与持久化:写入怎么不丢,崩溃怎么恢复
后端·mongodb·面试
葫芦和十三11 小时前
图解 MongoDB 16|压缩:snappy、zstd 和 zlib 的取舍
后端·mongodb·面试
祎雪双十Gy15 小时前
从 DataX 的配置加载说起:我用 FastJson2 做了一个轻量级动态配置管理库
java·后端
labixiong15 小时前
实现一个能跑的迷你版Promise(一)
前端·javascript·面试
小锋java123415 小时前
分享一套锋哥原创的SpringBoot4+Vue3宠物领养网站系统
java
考虑考虑18 小时前
Java实现hmacsha1加密算法
java·后端·java ee
掉鱼的猫19 小时前
Spring Boot → Solon 注解迁移实战指南:一张对照表说清楚
java·spring boot
plainGeekDev19 小时前
广播接收器 → Flow + Lifecycle
android·java·kotlin