每日一道面试题 10:synchronized 与 ReentrantLock 的核心区别及生产环境如何选型?

一、核心区别(面试必背,6 大维度清晰对比)

1. 底层实现

  • synchronized :JVM 层面实现,基于 Monitor 监视器锁,是 Java 内置关键字,由 JVM 自动加锁、释放锁。

  • ReentrantLock :JDK 层面实现,基于 AQS 队列同步器 ,是 java.util.concurrent.locks 包下的 API 类,需手动 lock() / unlock()

2. 可重入性

两者均支持可重入(同一个线程可重复获取锁,避免死锁)。

3. 锁获取与释放方式

  • synchronized :自动加锁,代码块执行完毕 / 抛出异常后自动释放锁,无需手动操作。
  • ReentrantLock :必须手动 lock() 加锁,unlock() 释放锁,且必须在 finally 中释放,否则极易死锁。

4. 公平锁支持

  • synchronized只支持非公平锁,线程竞争锁时随机抢占,无法保证顺序。
  • ReentrantLock :支持公平锁 & 非公平锁 ,构造方法传入 true 即为公平锁(按等待顺序获取锁)。

5. 高级特性

  • synchronized:无额外特性,仅支持基础互斥锁。
  • ReentrantLock :支持 限时等待锁可中断锁多个 Condition 条件队列,适合复杂同步场景。

6. 性能表现

  • JDK 1.6 以前:synchronized 性能远低于 ReentrantLock。
  • JDK 1.6 及以后:synchronized 引入偏向锁、轻量级锁、锁升级机制,高并发场景下性能与 ReentrantLock 接近

二、核心 API 与使用示例

1. synchronized 示例

java 复制代码
// 同步方法
public synchronized void method() {
    // 临界区代码
}

// 同步代码块
public void method() {
    synchronized (this) {
        // 临界区代码
    }
}

2. ReentrantLock 示例

java 复制代码
private Lock lock = new ReentrantLock();

public void method() {
    lock.lock();
    try {
        // 临界区代码
    } finally {
        // 必须在finally释放,避免异常导致锁泄露
        lock.unlock();
    }
}

三、适用场景

1.优先使用 synchronized

  1. 简单同步场景,代码简洁、无需手动释放锁,避免人为失误
  2. 无需公平锁、条件等待等高级特性。
  3. 追求代码简洁、减少样板代码。

2.优先使用 ReentrantLock

  1. 需要公平锁,保证线程按顺序获取锁。
  2. 需要限时获取锁tryLock(long timeout, TimeUnit unit)),避免无限阻塞。
  3. 需要可中断锁lockInterruptibly()),防止线程长时间卡死。
  4. 需要多个 Condition 条件队列,实现精准线程唤醒(如生产者消费者模式)。

四、面试高频追问

追问 1:什么是可重入锁?为什么要设计成可重入?

  • 可重入锁:同一个线程可多次获取同一把锁,不会自己锁死自己。
  • 目的:避免递归调用、多层嵌套同步代码时出现死锁。

追问 2:ReentrantLock 为什么必须在 finally 里 unlock?

  • 若临界区代码抛出异常,锁无法释放,会导致锁泄露,其他线程永久阻塞。

追问 3:synchronized 锁升级过程是什么?

无锁 → 偏向锁 → 轻量级锁 → 重量级锁竞争少时用轻量级锁(自旋),竞争激烈升级为重量级锁(阻塞)。

追问 4:Condition 相比于 synchronized 的 wait/notify 有什么优势?

  • synchronized 只有一个等待队列,唤醒时随机或全部唤醒。
  • ReentrantLock 可创建多个 Condition,实现分组等待、精准唤醒,避免无效唤醒,性能更高。

五、总结

synchronized 是 JVM 内置关键字,自动加解锁、使用简单;ReentrantLock 是 JDK API,支持公平锁、限时等待、多条件队列等高级特性。简单场景用 synchronized,复杂同步场景选 ReentrantLock。

相关推荐
014-code2 小时前
String.intern() 到底干了什么
java·开发语言·面试
摇滚侠3 小时前
JAVA 项目教程《苍穹外卖-12》,微信小程序项目,前后端分离,从开发到部署
java·开发语言·vue.js·node.js
楚国的小隐士3 小时前
为什么说Rust是对自闭症谱系人士友好的编程语言?
java·rust·编程·对比·自闭症·自闭症谱系障碍·神经多样性
春花秋月夏海冬雪3 小时前
代码随想录刷题 - 贪心Part1
java·算法·贪心·代码随想录
野生技术架构师4 小时前
2026年牛客网最新Java面试题总结
java·开发语言
Mr_Xuhhh4 小时前
深入理解Java抽象类与接口:从概念到实战
java·开发语言
wb043072014 小时前
使用 Java 开发 MCP 服务并发布到 Maven 中央仓库完整指南
java·开发语言·spring boot·ai·maven
Rsun045514 小时前
设计模式应该怎么学
java·开发语言·设计模式
5系暗夜孤魂4 小时前
系统越复杂,越需要“边界感”:从 Java 体系理解大型工程的可维护性本质
java·开发语言