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

详细聊聊 Synchronized,以及锁的升级过程

在Java中,synchronized关键字是用于实现线程同步的重要机制,它通过内置锁(Monitor)确保多个线程对共享资源的安全访问。


1. synchronized 的基本使用与实现原理

使用方式
  • 修饰实例方法:锁是当前对象实例。
    public synchronized void method() { ... }
    
  • 修饰静态方法:锁是当前类的Class对象。
    public static synchronized void staticMethod() { ... }
    
  • 同步代码块:需显式指定锁对象(任意对象均可)。
    synchronized (lockObject) { ... }
    
底层实现
  • Monitor 机制:每个对象关联一个Monitor(监视器锁),通过monitorentermonitorexit字节码指令实现加锁/解锁。
    • 线程进入同步块时,执行monitorenter尝试获取锁。
    • 退出同步块时,执行monitorexit释放锁。

2. 对象头与锁状态标记

每个Java对象在内存中分为三部分:对象头(Header)实例数据(Instance Data)对齐填充(Padding)。对象头中的Mark Word字段记录了锁状态信息。

Mark Word 结构(以64位JVM为例)
锁状态存储内容
无锁对象的哈希码、分代年龄、是否偏向锁(1 bit)
偏向锁偏向线程ID、偏向时间戳、分代年龄、锁标志位(01)
轻量级锁指向线程栈中锁记录(Lock Record)的指针,锁标志位(00)
重量级锁指向Monitor对象(重量级锁)的指针,锁标志位(10)

3. 锁的升级过程

JVM根据线程竞争情况动态调整锁状态,以减少性能开销。锁升级的路径为:
无锁 → 偏向锁 → 轻量级锁 → 重量级锁

(1) 偏向锁(Biased Locking)
  • 适用场景:单线程反复进入同步块,无实际竞争。
  • 核心机制
    • 对象首次被线程访问时,将线程ID写入Mark Word,进入偏向模式。
    • 后续该线程进入同步块时,无需执行CAS操作,直接检查线程ID是否匹配。
  • 优势:消除无竞争时的同步开销。
  • 撤销条件
    • 其他线程尝试获取锁时,触发偏向锁撤销。
    • 需要等待全局安全点(STW),检查原线程是否存活或已释放锁。
(2) 轻量级锁(Lightweight Locking)
  • 适用场景:多线程交替执行同步块,竞争不激烈。
  • 核心机制
    • 线程在栈帧中创建锁记录(Lock Record),将对象Mark Word复制到锁记录中。
    • 通过CAS将Mark Word替换为指向锁记录的指针。成功则获取锁;失败则膨胀为重量级锁。
  • 优势:避免线程阻塞,通过自旋(CAS)减少内核态切换开销。
  • 自旋优化
    • 适应性自旋:JVM根据历史自旋成功率动态调整自旋次数。
(3) 重量级锁(Heavyweight Locking)
  • 适用场景:多线程高并发竞争。
  • 核心机制
    • Monitor对象(C++实现)管理线程竞争,包含_owner(持有者)、_EntryList(阻塞队列)、_WaitSet(等待队列)。
    • 未获取锁的线程进入_EntryList,由操作系统调度(涉及用户态到内核态切换)。
  • 特点:线程阻塞,响应慢但公平。

4. 锁升级的触发条件

步骤触发条件
无锁 → 偏向锁对象首次被线程访问,JVM启用偏向锁(默认开启,Java 15后需手动开启)。
偏向锁 → 轻量级锁其他线程尝试获取锁,导致偏向锁撤销。
轻量级锁 → 重量级锁CAS自旋失败(超过阈值或竞争激烈),触发锁膨胀(Inflate)。

5. 锁的不可逆性与性能权衡

  • 不可逆性:锁升级后无法降级,因为降级会增加复杂性和性能损耗。
  • 性能权衡
    • 偏向锁:适合单线程场景,但撤销成本高(需STW)。
    • 轻量级锁:适合低竞争场景,依赖CAS自旋。
    • 重量级锁:适合高竞争场景,牺牲响应时间保证稳定性。

6. Monitor 的详细结构

Monitor对象(如ObjectMonitor)包含以下关键字段:

  • _owner:当前持有锁的线程。
  • _recursions:锁的重入次数。
  • _EntryList:等待获取锁的线程队列。
  • _WaitSet:调用wait()后进入等待状态的线程队列。

7. 实际案例:锁升级过程

场景:两个线程交替执行同步块
  1. 初始状态:对象无锁。
  2. 线程A进入同步块:升级为偏向锁,Mark Word记录线程A的ID。
  3. 线程B尝试进入:触发偏向锁撤销,升级为轻量级锁,线程B通过CAS竞争。
  4. 线程B CAS失败:自旋后仍未成功,升级为重量级锁,线程B进入_EntryList阻塞。

8. 最佳实践

  • 避免过度同步:减少锁粒度(如使用ConcurrentHashMap)。
  • 优先使用轻量级工具:如ReentrantLockStampedLock(需手动管理)。
  • 监控锁竞争:通过JVM参数(-XX:+PrintFlagsFinal)或工具(Arthas)分析锁状态。

在这里插入图片描述

http://www.xdnf.cn/news/4647.html

相关文章:

  • 二极管的动态特性
  • AI(学习笔记第二课) 使用langchain进行AI开发
  • Coco AI 开源应用程序 - 搜索、连接、协作、您的个人 AI 搜索和助手,都在一个空间中。
  • 【CTFer成长之路】举足轻重的信息搜集
  • 数据结构之串
  • 【PmHub后端篇】PmHub Gateway全局过滤器:接口调用耗时统计及黑白名单配置技术深度解析
  • 57.[前端开发-前端工程化]Day04-webpack插件模式-搭建本地服务器
  • XML语言
  • 企业开发平台大变革:AI 代理 + 平台工程重构数字化转型路径
  • Android单例模式知识总结
  • 02_JVM
  • Mockoon 使用教程
  • 为什么使用Less替代原始CSS?
  • 学习黑客MAC 地址
  • 数字孪生市场格局生变:中国2025年规模214亿,工业制造领域占比超40%
  • 安卓应用卡顿、性能低下的背后原因
  • 【文献阅读】Depth Anything Unleashing the Power of Large-Scale Unlabeled Data
  • 2025-05-08 Unity 网络基础9——FTP通信
  • Linux的基础开发工具
  • 手机上使用的记录笔记的软件推荐哪一款
  • SAP 交货单行项目含税金额计算报cx_sy_zerodivide处理
  • 云手机虚拟地址技术的运营场景
  • n8n - 开放灵活的智能自动化工作流平台
  • uniapp自定义步骤条(可二开进行调试)
  • shader中性能优化
  • docker 部署clickhouse
  • App Store支付新政重构跨境电商生态:eBay卖家的突围之道
  • vue中scss使用js的变量
  • OpenCv实战笔记(3)基于opencv实现调用摄像头并实时显示画面
  • 【WEB3】区块链、隐私计算、AI和Web3.0——隐私计算(2)