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

Java 中的锁机制详解

Java 中的锁机制是实现多线程并发控制的核心手段,用于保证临界资源在多线程访问时的安全性。锁的设计与实现主要依赖 JDK 提供的 synchronizedjava.util.concurrent.locks 包。


一、锁的分类总览

分类维度锁类型
实现层面Java 内置锁(synchronized)
JUC 显式锁(ReentrantLock 等)
可重入性可重入锁 / 非可重入锁
公平性公平锁 / 非公平锁
读写粒度独占锁 / 共享锁(如 ReadWriteLock)
乐观与悲观乐观锁(CAS) / 悲观锁
锁粒度与升级偏向锁 → 轻量级锁 → 重量级锁
其他特性自旋锁、可中断锁、可定时锁

二、Java 常见锁类型详解

1. synchronized(内置锁)

特点:
  • 自动获取和释放锁
  • 支持可重入阻塞式重量级
  • 编译器层面支持,底层使用 monitor(对象监视器)
使用方式:
synchronized(obj) {// 临界区
}

或用于方法:

public synchronized void doSomething() {}
适用场景:
  • 代码简单的同步场景
  • 不需要手动加解锁的逻辑
  • 并发竞争不激烈的情况下性能较好(配合锁优化)

2. ReentrantLock(显示锁)

特点:
  • 可重入
  • 支持中断锁定时锁公平锁条件变量
  • 手动加锁与释放(需 finally 中 unlock)
使用方式:
ReentrantLock lock = new ReentrantLock();lock.lock();
try {// 临界区
} finally {lock.unlock();
}
适用场景:
  • 需要更高级控制功能(如公平性、可中断、条件变量)
  • 更灵活地配合条件对象实现等待/通知机制

3. ReadWriteLock(读写锁)

特点:
  • 提供读锁(共享)和写锁(独占)
  • 支持多个线程并发读,写操作需互斥
使用方式:
ReadWriteLock rwLock = new ReentrantReadWriteLock();
Lock readLock = rwLock.readLock();
Lock writeLock = rwLock.writeLock();
适用场景:
  • 读多写少的缓存类、配置类
  • 提高读并发性能

4. StampedLock(JDK 8 新增)

特点:
  • 不可重入
  • 三种模式:写锁、悲观读锁、乐观读锁
  • 适用于对性能要求极高的读写场景
示例:
long stamp = stampedLock.tryOptimisticRead();
try {// 读操作if (!stampedLock.validate(stamp)) {// fallback to pessimistic}
} finally {stampedLock.unlock(stamp);
}
适用场景:
  • 高并发读访问
  • 优化 CPU 缓存一致性开销(如 LRU 缓存)

5. 乐观锁(CAS)

特点:
  • 不阻塞,基于版本号或原子变量进行更新
  • 实现机制依赖 Unsafe.compareAndSwapXxx
使用场景:
  • 原子类:AtomicInteger, AtomicLong
  • 非常适合频繁读、偶尔写的场景,如计数器、非阻塞队列

6. 自旋锁

特点:
  • 在获取不到锁时,不立即挂起线程,而是循环尝试
  • 减少线程上下文切换,但可能造成 CPU 空转
应用场景:
  • 锁等待时间非常短的场景,如 CPU 计算密集型任务

三、锁升级过程(JVM 优化)

JVM 会根据竞争情况自动将锁从低成本升级为高成本锁:

偏向锁 → 轻量级锁 → 重量级锁
锁类型描述
偏向锁只有一个线程竞争,自动偏向该线程
轻量级锁多线程竞争但没有阻塞
重量级锁多线程激烈竞争,发生阻塞

四、常见锁的比较表

锁类型可重入阻塞公平可选中断响应性能
synchronized
ReentrantLock
ReadWriteLock高(读多)
StampedLock极高
乐观锁(CAS)极高

五、典型应用场景总结

场景类型推荐锁方案
简单线程同步synchronized
复杂并发控制ReentrantLock
读多写少ReadWriteLock / StampedLock
频繁原子操作原子类(CAS)
高性能读缓存StampedLock 乐观读
线程间通知机制Lock + Condition
秒杀、抢单、库存分布式锁(Redisson 等)

附录:锁死与锁优化

死锁四大必要条件

  1. 互斥使用
  2. 不可抢占
  3. 请求与保持
  4. 循环等待

锁优化方向

  • 减少锁粒度
  • 减少锁持有时间
  • 使用无锁 / CAS 替代
  • 使用读写锁区分读写冲突

总结

Java 的锁机制涵盖了从语言级别到并发包的全套支持,配合 JVM 锁优化(偏向、轻量、重量级锁),可以根据业务并发需求灵活选型。

锁的选型应关注:

  • 线程竞争激烈程度
  • 可读可写并发比例
  • 是否需要中断或公平性控制
  • 对性能的敏感度
http://www.dtcms.com/a/266231.html

相关文章:

  • 【HarmonyOS Next之旅】DevEco Studio使用指南(四十) -> 灵活定制编译选项
  • Kotlin 安装使用教程
  • 深度剖析:如何解决Node.js中mysqld_stmt_execute参数错误
  • JVM类加载系统详解:深入理解Java类的生命周期
  • 数字资产革命中的信任之锚:RWA法律架构的隐形密码
  • 基于Linux的Spark本地模式环境搭建实验指南
  • 白色氧化铈:“白”光之下的科技之美
  • 衡石科技破解指标管理技术难题:语义层建模如何实现业务与技术语言对齐?
  • 【C#】命名空间
  • 尝试安装使用无头cms strapi (未完成)
  • 【数据结构】时间复杂度与空间复杂度
  • 【C++】访问者模式中的双重分派机制详解
  • 淋巴细胞激活靶点CD6
  • 【人工智能与机器人研究】优化YOLOv11模型:基于多尺度注意力机制的小目标检测性能提升研究
  • RRF (Reciprocal Rank Fusion) 排序算法详解
  • 【排序算法】
  • Vue3封装动态Form表单
  • 第二章-AIGC入门-开启AIGC音频探索之旅:从入门到实践(6/36)
  • 【学术写作+AI实战】株洲高校科研写作研修班全纪实:核心期刊编辑与AI专家的联合授课笔记
  • Web前端数据可视化:ECharts高效数据展示完全指南
  • 【JavaEE】计算机工作原理
  • JavaEE初阶第七期:解锁多线程,从 “单车道” 到 “高速公路” 的编程升级(五)
  • 运维打铁:企业云服务解决方案
  • openEuler 24.03 全流程实战:用 Ansible 5 分钟部署分布式 MinIO 高可用集群
  • Django+DRF 实战:从异常捕获到自定义错误信息
  • 深度分析:Microsoft .NET Framework System.Random 的 C++ 复刻实现
  • 切出idea窗口自动编译,关闭idea自动编译
  • WPF+HelixToolkit打造炫酷自定义3D贴图立方体盒子模型
  • 机器学习在智能供应链中的应用:需求预测与物流优化
  • Java技术深潜:从并发陷阱到云原生突围