当前位置: 首页 > news >正文

Java中常见的锁synchronized、ReentrantLock、ReentrantReadWriteLock、StampedLock

在Java中,锁是实现多线程同步的核心机制。不同的锁适用于不同的场景,理解其实现原理和使用方法对优化性能和避免并发问题至关重要。


一、隐式锁:synchronized 关键字

实现原理
  • 基于对象监视器(Monitor):每个Java对象都有一个内置的监视器锁(monitor lock),通过 synchronized 关键字获取。
  • 锁升级机制(JVM优化):
    • 偏向锁:无竞争时,标记线程ID,避免CAS操作。
    • 轻量级锁:通过CAS竞争锁,失败后升级为重量级锁。
    • 重量级锁:通过操作系统互斥量(mutex)实现线程阻塞。
使用方法
// 1. 同步代码块
synchronized (obj) { // 临界区代码
}// 2. 同步实例方法
public synchronized void method() { }// 3. 同步静态方法
public static synchronized void method() { }
适用场景
  • 简单同步需求:无需复杂锁功能的场景(如可中断、超时等)。
  • 代码简洁性优先:自动释放锁,避免忘记解锁的风险。

二、显式锁:ReentrantLock

实现原理
  • 基于AQS(AbstractQueuedSynchronizer)
    • 通过 state 变量(CAS操作)记录锁状态。
    • 使用CLH队列管理等待线程。
  • 支持公平性:可选择公平锁(按排队顺序获取)或非公平锁(插队竞争)。
使用方法
ReentrantLock lock = new ReentrantLock();
lock.lock();
try {// 临界区代码
} finally {lock.unlock(); // 必须手动释放
}// 高级功能示例:尝试获取锁
if (lock.tryLock(1, TimeUnit.SECONDS)) {try { /* ... */ } finally { lock.unlock(); }
}
适用场景
  • 复杂锁需求:需要可中断、超时、公平性等特性。
  • 细粒度控制:如跨方法加锁解锁(synchronized 只能在代码块内)。

三、读写锁:ReentrantReadWriteLock

实现原理
  • 分离读锁(共享)和写锁(独占)
    • 读锁允许多线程并发读,写锁独占。
    • AQS的 state 高16位记录读锁,低16位记录写锁。
使用方法
ReadWriteLock rwLock = new ReentrantReadWriteLock();
Lock readLock = rwLock.readLock();
Lock writeLock = rwLock.writeLock();// 读操作
readLock.lock();
try { /* 读数据 */ } finally { readLock.unlock(); }// 写操作
writeLock.lock();
try { /* 写数据 */ } finally { writeLock.unlock(); }
适用场景
  • 读多写少:如缓存系统、高频查询场景。
  • 数据一致性要求:写操作需要互斥,读操作可并发。

四、乐观锁:StampedLock(Java 8+)

实现原理
  • 基于票据(Stamp)的锁机制
    • 支持三种模式:写锁、悲观读锁、乐观读。
    • 乐观读不阻塞写操作,通过验证Stamp判断数据一致性。
使用方法
StampedLock stampedLock = new StampedLock();// 乐观读
long stamp = stampedLock.tryOptimisticRead();
// 读取数据
if (!stampedLock.validate(stamp)) {// 数据被修改,升级为悲观读锁stamp = stampedLock.readLock();try { /* 重新读取数据 */ } finally { stampedLock.unlockRead(stamp); }
}// 写锁
long stamp = stampedLock.writeLock();
try { /* 写数据 */ } finally { stampedLock.unlockWrite(stamp); }
适用场景
  • 读多写少且容忍数据不一致:如统计、日志处理。
  • 极高性能需求:乐观读避免锁竞争,但需处理验证逻辑。

五、其他锁机制

1. Condition 条件变量
  • ReentrantLock 配合使用,实现线程间协作(类似 wait/notify)。
  • 典型场景:生产者-消费者模型。
2. 分布式锁
  • 如基于Redis的 Redisson 或ZooKeeper实现。
  • 适用场景:跨JVM或分布式系统同步。

六、锁的选择与性能优化

锁对比表
锁类型特性性能适用场景
synchronized自动释放,非公平锁低竞争时高效简单同步需求
ReentrantLock可中断、超时、公平锁高竞争时高效复杂锁需求
ReadWriteLock读写分离读多写少高效缓存、查询系统
StampedLock乐观读,支持锁升级极高并发读多写少,容忍数据不一致
最佳实践
  1. 减少锁粒度:缩小临界区范围。
  2. 避免嵌套锁:防止死锁(如按固定顺序获取锁)。
  3. 监控锁竞争:使用JProfiler或JStack分析锁状态。

七、总结

  • 简单场景优先选择 synchronized(JVM优化成熟)。
  • 复杂需求使用 ReentrantLockReadWriteLock
  • 极致性能考虑 StampedLock,但需谨慎处理数据一致性。

合理选择锁类型,结合性能测试和监控,是构建高效并发系统的关键。

相关文章:

  • 【机器学习】朴素贝叶斯算法:原理剖析与实战应用
  • 深度补全网络:如CSPN++填补稀疏点云的深度信息
  • 修改 <li> 元素小圆点的颜色
  • 不连续数据区间天数累计sql
  • 手机投屏到电视方法
  • MongoDB导出和导入数据
  • 【大疆dji】边缘计算模块在大疆机场中的位置
  • Datawhale AI春训营】AI + 新能源(发电功率预测)Task1
  • nohup的使用
  • 2025年第16届蓝桥杯嵌入式竞赛学习笔记(十四):RTC实时时钟
  • ESB —— 企业集成架构的基石:功能、架构与应用全解析
  • 详细解释浏览器是如何渲染页面的?
  • 国网B接口协议图像数据上报通知接口流程详解以及上报失败原因(电网B接口)
  • Docker 网络详解:从 docker0 网桥到网络命名空间
  • 深入Docker核心技术:从Namespace到容器逃逸防御
  • OpenCV 04.19 练习
  • Python带有else子句的循环语句
  • 【漫话机器学习系列】210.标准化(Standardization)
  • docker 大模型
  • Cribl 优化EC2 ip-host-region 数据
  • 普京签署法律,诋毁俄军将面临最高7年监禁
  • 广西出现今年首场超警洪水
  • 东北三省,十年少了一个“哈尔滨”
  • 圆桌|耐心资本对科技创新有何意义?天使投资最关注哪些要素?
  • “万人大院”重组,上海交大校长丁奎岭:人才培养事关生存发展,再难也要改
  • 安徽省合肥市人大常委会原副主任杜平太接受审查调查