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

Java中synchronized的偏向锁是如何减少锁开销的

偏向锁(Biased Locking)是一种优化 Java synchronized 锁的机制,旨在减少在无竞争情况下的锁开销。它通过将锁偏向于单个线程来优化锁的性能。以下是偏向锁减少锁开销的具体方式和原理:

偏向锁的工作原理

  1. 锁的初始状态:

    • 当一个对象的锁首次被持有时,JVM 会将该对象的锁标记为偏向锁,并将对象的头部中的 Mark Word(对象头的一部分)设置为偏向于当前线程。此时,只有这个线程可以获得锁,而其他线程则不能获取。
  2. 偏向锁的记录:

    • 对象头中的 Mark Word: 偏向锁的状态保存在对象头的 Mark Word 中。Mark Word 中会保存持有锁的线程 ID。如果锁的持有者线程进行锁操作,那么在该线程访问该锁对象时,JVM 可以快速检查到该线程已经持有锁,无需进行额外的同步操作。
  3. 锁的获取和释放:

    • 获取锁: 当线程尝试获取锁时,JVM 会检查对象头中的 Mark Word。如果 Mark Word 中记录的线程 ID 与当前线程匹配,则说明当前线程已经持有锁,此时锁的获取是“无成本”的。
    • 释放锁: 当持有锁的线程释放锁时,JVM 会将对象头中的 Mark Word 恢复为初始状态,使得其他线程可以重新获取锁。
  4. 撤销偏向锁:

    • 锁竞争: 如果偏向锁的持有者线程在持有锁期间被中断或其他线程尝试获取该锁,那么偏向锁会被撤销。此时,锁会升级为轻量级锁(Lightweight Locking),并且偏向锁的记录会被移除。

偏向锁减少锁开销的方式

  1. 减少同步操作的开销:

    • 减少标记和检查: 偏向锁在没有竞争的情况下,锁的获取和释放不会进行额外的同步操作。通过直接检查对象头中的 Mark Word,JVM 能够快速判断是否需要进行加锁操作。
  2. 避免锁的重入:

    • 线程 ID 的记录: 偏向锁记录了持有锁的线程 ID,当持有锁的线程再次尝试获取锁时,可以快速确认锁已经被持有,无需进行实际的加锁和解锁操作,从而减少了锁的开销。
  3. 减少线程调度开销:

    • 避免上下文切换: 偏向锁在没有竞争的情况下,避免了线程之间的上下文切换,从而减少了由于线程切换导致的性能开销。

偏向锁的局限性

  1. 锁竞争:

    • 竞争撤销: 如果有多个线程同时竞争一个偏向锁,偏向锁会被撤销并升级为轻量级锁,这会引入额外的开销。此时,锁的优化效果可能会减少。
  2. 长时间持有:

    • 长时间持有偏向锁: 如果线程长时间持有偏向锁而没有竞争,虽然可以减少锁的开销,但在线程进行操作时,偏向锁的持有者需要保持对象头中的线程 ID,这可能会导致一些内存开销。

示例代码

以下示例演示了如何触发偏向锁的机制:

public class BiasedLockExample {private static final Object lock = new Object();public static void main(String[] args) {// 创建多个线程来访问同步方法Runnable task = () -> {synchronized (lock) {System.out.println(Thread.currentThread().getName() + " - Acquired lock");try {Thread.sleep(100); // 模拟工作} catch (InterruptedException e) {e.printStackTrace();}}};// 启动多个线程来测试偏向锁的效果for (int i = 0; i < 5; i++) {new Thread(task, "Thread-" + i).start();}}
}

在这段代码中,synchronized 方法的对象 lock 初始会使用偏向锁。随着线程的竞争,JVM 会将偏向锁升级为轻量级锁或重量级锁(如有必要)。

总结

  • 偏向锁: 主要用于减少在没有锁竞争时的开销。
  • 对象头中的 Mark Word: 用于记录持有锁的线程 ID,从而优化锁的获取和释放。
  • 锁竞争和撤销: 当检测到锁竞争时,偏向锁会被撤销并升级为轻量级锁,优化锁的性能。

偏向锁通过在没有竞争的情况下减少同步开销,从而提高程序的执行效率。

http://www.lryc.cn/news/424276.html

相关文章:

  • react18 + ts 使用video.js 直播.m3u8格式的视频流
  • 使用 onBeforeRouteLeave 组合式函数提升应用的用户体验
  • uni-app 吸顶方案总结
  • 【C#】知识汇总
  • 1、Unity【基础】3D数学
  • 虚拟机ubuntu22的扩容记录
  • Docker 常用配置
  • 通过示例了解 .NET Core 中的依赖注入
  • fetch、FormData上传多张图片
  • C++STL详解(五)——list类的具体实现
  • 鸿蒙(API 12 Beta3版)【使用投播组件】案例应用
  • 【STM32项目】在FreeRtos背景下的实战项目的实现过程(一)
  • C#垃圾处理机制相关笔记
  • C语言memcmp函数
  • 低代码: 组件库测试之Vue环境下的测试工具以及测试环境搭建
  • 【Vue3】高颜值后台管理模板推荐
  • 详细介绍Pytorch中torchvision的相关使用
  • AI部署——主流模型推理部署框架
  • PyTorch之loading fbgemm.dll异常的解决办法
  • Vscode——如何实现 Ctrl+鼠标左键 跳转函数内部的方法
  • 力扣热题100_回溯_78_子集
  • 浏览器如何工作(一)进程架构
  • 【LeetCode】两数之和
  • UE5学习笔记11-为拿取武器添加动画
  • 68. 文本左右对齐【 力扣(LeetCode) 】
  • 【中等】 猿人学web第一届 第6题 js混淆-回溯
  • 低、中、高频率段具体在不同应用中的范围是多少
  • Oxford Model600 Model400低温氦压缩机cryogenic helium compressor手侧
  • Golang面试题四(并发编程)
  • 计算机学生高效记录并整理编程学习笔记的方法