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

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.xdnf.cn/news/14814.html

相关文章:

  • 服装零售企业跨区域运营难题破解方案
  • Kotlin 安装使用教程
  • SAP SD模块之业务功能剖析
  • CI/CD持续集成与持续部署
  • 开源 vGPU 方案 HAMi: corememory 隔离测试
  • 深度剖析:如何解决Node.js中mysqld_stmt_execute参数错误
  • Java 数据类型与变量
  • Oracle如何使用序列 Oracle序列使用教程
  • OpenCV中DPM(Deformable Part Model)目标检测类cv::dpm::DPMDetector
  • KVM高级功能部署
  • Go应用容器化完全指南:构建最小化安全镜像的终极实践
  • 【MySQL\Oracle\PostgreSQL】迁移到openGauss数据出现的问题解决方案
  • Python入门Day2
  • Python字符与ASCII转换方法
  • Qt 事件
  • Python从入门到精通——第一章 Python简介
  • 从 TCP/IP 协议栈角度深入分析网络文件系统 (NFS)
  • join性能问题,distinct和group by性能,备库自增主键问题
  • 孪生素数猜想 - 张益唐的核心贡献和陶哲轩的改进
  • vue-37(模拟依赖项进行隔离测试)
  • 互联网大厂Java面试实录:Spring Boot与微服务在电商场景中的应用
  • 经典灰狼算法+编码器+双向长短期记忆神经网络,GWO-Transformer-BiLSTM多变量回归预测,作者:机器学习之心!
  • List中的对象进行排序处理
  • Go基础(Gin)
  • Python 机器学习核心入门与实战进阶 Day 1 - 分类 vs 回归
  • 扣子空间PPT生产力升级:AI智能生成与多模态创作新时代
  • 【Linux仓库】进程优先级及进程调度【进程·肆】
  • Linux之Socket编程Tcp
  • Spring Cloud(微服务部署与监控)
  • Superman