「JVM 高效并发」锁优化

news2025/1/13 10:33:59

为了线程间更高效的共享数据及解决竞争问题,提高程序执行效率,JDK 6 做了大量锁优化,如适应性自旋Adaptive Spinning)、锁消除Lock Elimination)、锁膨胀Lock Coarsening)、轻量级锁Lightweight Locking)、偏向锁Biased Locking)等;

文章目录

      • 1. 自旋锁与自适应自旋
      • 2. 锁消除
      • 3. 锁粗化
      • 4. 轻量级锁
      • 5. 偏向锁

1. 自旋锁与自适应自旋

  • 自旋锁,让后面请求锁的线程稍等一会,但不放弃处理器的执行时间,看持有锁的线程是否很快释放锁;为了让线程等待,需让线程执行一个忙循环(自旋);

自旋等待不能代替阻塞,它虽然避免了线程切换的开销,但会占用处理器时间;锁被占用的时间很短时自旋等待效果很好,但长时间自旋的线程会浪费 CPU 资源;

-XX:PreBlockSpin,设置自旋次数,超过后使用传统方式挂起线程;

  • 自适应自旋,自旋时间不是固定的,而是由前一次在同一锁上自旋时间及锁的拥有者的状态来决定(上次自旋成功获得锁,则可能允许自旋等待更长的时间;上次自旋失败,则可能不在允许自旋等待);

2. 锁消除

  • 锁消除,JVM 在即时编译期对一些代码要求同步,但被检测到不可能存在共享数据竞争(逃逸分析,堆上所有数据都不会逃逸出线程,就可以把它们当做栈上数据对待,即线程私有)的锁进行消除;

JDK 5 之前同步代码示例

public String concatString(String s1, String s2, String s3) {
    return s1 + s2 + s3;
}

javac 编译后的连接操作

public String concatString(String s1, String s2, String s3) {
    StringBuffer sb = new StringBuffer();
    sb.append(s1);
    sb.append(s2);
    sb.append(s3);
    return sb.toString();
}

每个 append() 方法内部都是一个同步块,锁对象是 sb;经过逃逸分析,发现 sb 的所有引用都不会逃逸出 concatString() 方法,其他线程无法访问到它,因此可以在即时编译阶段安全的消除锁;(在解释执行存在锁,在编译执行没有锁;JDK 5 之后改用非线程安全的 StringBuilder 就不会自动加锁了);

3. 锁粗化

原则上编写代码时,推荐奖同步块的作用范围限制得尽量小,只有共享数据的实际作用域才进行同步,这样即使存在锁竞争,也可尽可能快地拿到锁;

但若频繁对同一个对象反复加锁和解锁,甚至在循环体上加锁,即使没有线程竞争,频繁互斥操作也会导致不必要的性能损耗;因此 JVM 会直接把加锁同步的范围扩展(粗化)到整个操作序列的外部;(如上例,直接在第一个 append() 之前加锁,在最后一个 append() 之后解锁);

4. 轻量级锁

不是用来代替重量级锁的,而是在没有多线程竞争时,减少传统重量级锁的操作系统互斥产生的性能消耗;

请添加图片描述

在线程即将进入同步块时,若此同步对象还没有被锁定(01 状态),JVM 会先在当前线程的栈帧中建立一个锁记录(Lock Record)空间,用于存储锁对象目前 Mark Work 的拷贝(Displace Mark Word);

然后 JVM 尝试使用 CAS 操作把对象的 Mark Word 更新为执行 Lock Record 的指针;若 CAS 操作成功,代表该线程拥有了这个对象的锁(对象 Mark Word 的锁标志为将转变为 00 状态);

若 CAS 操作失败,代表至少有一个线程与当前线程竞争获取该对象的锁,JVM 会先检查对象的 Mark Word 是否指向当前线程的栈帧;若是,说明当前线程已拥有这个对象的锁,直接进入同步块继续执行即可;若否,则说明这个对象的锁已经被其他线程占有,若出现两条以上线程占用同一个锁,则轻量级锁不再有效,必须膨胀为重量级锁,锁标志状态变为 10(此时 Mark Word 中存储的是指向重量级锁,互斥量的指针),等待锁的线程必须进入阻塞状态;

轻量级锁的解锁过程同样使用 CAS 操作,若对象的 Mark Word 任然指向线程的锁记录,就用 CAS 操作把对象当前的 Mark Word 和线程中复制的 Displaced Mark Word 替换回来;若成功,则整个同步过程顺利完成,否则说明有其他线程尝试过获取锁,就要释放锁并唤醒被挂起的线程;

轻量级锁的适用场景是:绝大部分锁在整个同步周期内部存在竞争;没有竞争的情况下,轻量级锁通过 CAS 操作可以成功避免使用互斥量的开销;但若竞争存在,除了互斥量本身的开销,额外的 CAS 操作开销会让消耗比传统重量级锁更大;

5. 偏向锁

JDK 6 引入的锁优化措施;用于消除数据在无竞争情况下的同步原语;(轻量级锁是消除无竞争状态下的互斥量);

如果第一个获得锁的线程在接下来的执行中一致没有其他线程竞争,则持有偏向锁,线程永远不需要再进行同步;

锁对象第一次被线程获取时,JVM 会把对象头中的标志位设置为 01,把偏向模式设置为 1,表示进入偏向模式;同时使用 CAS 操作把获得这个锁的线程的 ID 记录在对象的 Mark Word 中,若 CAS 操作成功,持有偏向锁的线程以后每次进入这个锁相关的同步块都不再进行同步操作(加锁、解锁、Mark Word 更新操作等);

一旦出现另外的线程去尝试获取这个锁,偏向模式立刻结束;若对象未被锁定,撤销偏向(偏向模式置为 0)后标志位恢复到未锁定(01),若对象已被锁定,则标志位置为 00(使用轻量级锁);

请添加图片描述

当一个对象已经计算过一次一致性哈希码,它就无法再进入偏向锁状态;而当一个对象正处理偏向锁状态,又收到计算一致性哈希码请求时,它的偏向状态会立即测校,且锁会膨胀为重量级锁,对象头指向重量级锁的位置(代表重量级锁的 ObjectMonitor 类里可以记录非加锁状态下的 Mark Word,也可以存储原来的哈希码);

若程序中大多数的锁总是被多个不同线程访问,偏向模式就是多余的,可能使用 -XX:-UseBiasedLocking 禁用偏向锁优化反而可使性能提升;

  • -XX:+UseBiasedLocking,启用偏向锁,JDK 6 开始 HotSpot VM 默认开启;

上一篇:「JVM 高效并发」线程安全
专栏:《JVM 体系梳理》

PS:感谢每一位志同道合者的阅读,欢迎关注、评论、赞!


参考资料:

  • [1]《深入理解 Java 虚拟机》

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/379458.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

2、监控界面设计

【任务描述】本任务要求使用相对布局或约束布局以及相应的控件完成智慧园区监控系统界面开发一、相对布局(RelativeLayout)概述相对布局(RelativeLayout)是一种根据父容器和兄弟控件作为参照来确定控件位置的布局方式。使用相对布…

《机器学习》- 习题解析 - 第一章

《机器学习》- 习题 - 第一章 文章目录《机器学习》- 习题 - 第一章一、示例-计算表1.1中的版本空间二、习题 1 - 计算题目中的版本空间三、单个合取式&析合范式的概念四、习题 2 - 计算题目中假设空间的规模大小一、示例-计算表1.1中的版本空间 首先从概念上理解版本空间…

一起玩转开源数据库!OceanBase DevCon 之开源生态全景解析

​ 2023 年 3 月 25 日,首次 OceanBase 开发者大会将在北京举办,OceanBase 首席科学家阳振坤与 OceanBase CTO 杨传辉领携众多技术专家,将与开发者共同探讨单机分布式、云原生、HTAP 等数据库前沿趋势,OceanBase 开源技术全景生…

数据库——1.数据库设计的三大范式

这篇文章我们主要来讲一下数据库设计的三大范式,这个还是很有用的。 目录 1.概述 2.第一范式 3.第二范式 4.第三范式 5.小结 1.概述 为了建立冗余较小、结构合理的数据库,设计数据库时必须遵循一定的规则。在关系型数据库中这种规则就称为范式。范…

Python每日一练(20230301)

目录 1. 只出现一次的数字 2. 以特殊格式处理连续增加的数字 3. 最短回文串 1. 只出现一次的数字 给定一个非空整数数组,除了某个元素只出现一次以外,其余每个元素均出现两次。找出那个只出现了一次的元素。 说明: 你的算法应该具有线性…

「TCG 规范解读」基础设施架构和协议 (2)

可信计算组织(Ttrusted Computing Group,TCG)是一个非盈利的工业标准组织,它的宗旨是加强在相异计算机平台上的计算环境的安全性。TCG于2003年春成立,并采纳了由可信计算平台联盟(the Trusted Computing Platform Alli…

有什么好用的在线统计表单吗?

有什么好用的在线统计表单吗?最好是免费的?市面上这样的表单工具其实很多,先来看看题主的需求: 收集信息,数据统计数据分析,报表展示 以简道云在线表单为例,能完美实现题主这两个需求—— http…

携程面经1

面经 HDFS读写流程 1.读流程 客户端向NameNode发起读请求(如果存在)NameNode返回一批block地址客户端与第一个block的拓扑距离最近的节点建立连接以packet(64kb)的单位读取数据块。一个block读取完成后客户端会断开与该DataNod…

算法训练营 day59 动态规划 两个字符串的删除操作 编辑距离

算法训练营 day59 动态规划 两个字符串的删除操作 编辑距离 两个字符串的删除操作 583. 两个字符串的删除操作 - 力扣(LeetCode) 给定两个单词 word1 和 word2 ,返回使得 word1 和 word2 相同所需的最小步数。 每步 可以删除任意一个字符…

DBeaver连接mysql数据库图文教程

文章目录前言一、DBeaver连接mysql数据库二、文档下载地址前言 DBeaver是免费、开源、通用数据库工具,是许多开发开发人员和数据库管理员的所选。下面详细介绍Dbeaver连接mysql数据库的过程。 一、DBeaver连接mysql数据库 1、 打开Dbeaver后,按下图操…

【Unity】P4 脚本文件(基础)

Unity脚本文件(基础)适配的C#代码编辑器如何添加一个脚本文件获取蘑菇当前位置基础代码改变物体位置帧与帧更新前言 上一篇博文主要围绕Unity Inspector部分,围绕组件,资源文件,父子节点部分做介绍。 链接:…

阿里黑客入门学习资料流出来了!!

各位粉丝朋友大家好,最近看到很多粉丝朋友给我留言,希望我给大家找一些学习内容。前段时间整理了我平时常看的一些黑客相关的技术书籍,这些内容从未对外公开,今天分享给大家 ! 内容非常详细且全面,覆盖了W…

5分钟轻松拿下Java枚举

文章目录一、枚举(Enum)1.1 枚举概述1.2 定义枚举类型1.2.1 静态常量案例1.2.2 枚举案例1.2.3 枚举与switch1.3 枚举的用法1.3.1 枚举类的成员1.3.2 枚举类的构造方法1)枚举的无参构造方法2)枚举的有参构造方法1.3.3 枚举中的抽象方法1.4 Enum 类1.4.1 E…

c++系列12:使用vscode进行编译

1. 入门 1.1 操作方法 1)下载安装vscode 2)在扩展中搜索c/c extension pack并安装(或者直接打开cpp文件,会自动提示进行安装) 3)创建项目目录,会自动生成.vscode文件夹,里面是编译…

虹科分享 | Domo零售行业商业智能白皮书:《从零售企业的数据中获取价值》

市场因素、技术创新和不断增长的客户期望,给电子商务带来了新的机遇,与此同时也给传统零售行业带来了压力。零售业正面临着新的挑战:不断变化的需求模式和渠道、不断变化的服务期望、复杂的库存以及交付问题。为了解决这些问题,零…

Linux系统介绍及熟悉Linux基础操作

一、什么是Liunx Linux,全称GNU/Linux,是一种免费使用和自由传播的类UNIX操作系统,其内核由林纳斯本纳第克特托瓦兹(Linus Benedict Torvalds)于1991年10月5日首次发布,它主要受到Minix和Unix思想的启发&am…

机器学习知识总结 —— 21. 什么是主成分分析

文章目录什么是PCA(Principal Component Analysis)协方差矩阵什么是协方差协方差矩阵特征值与特征向量PCA降维什么是PCA(Principal Component Analysis) 在机器学习中,PCA(Principal Component Analysis&a…

除了Confluence,还有哪些好用的文档管理软件?测评

在早期,文档管理软件主要是为了将企业内部海量的电子文档集中存储、管理,通过设置共享权限进行内部员工的文档分发,有些甚至可能要提供API接口,便于将ERP、OA等系统的文档纳入其中,形成企业文档管理中心。而随着时间的…

window下的快捷程序链怎么设置环境变量|cmd直接运行快捷方式

对于需要在命令行执行的程序,每次都需要设置环境变量很是麻烦,而且也会导致非必要的文件也在环境变量里并且如果多版本共存软件也会导致只能一个存在环境变量里不然会冲突,这时候如果可以通过快捷方式那不就完美解决了么? 快捷方…

一文带你入门Docker

目录一、什么是Docker?1、背景2、Docker三要素3、Docker四个组成部分二、Docker安装步骤1、VM虚拟机下载2、centrOS 8下载3、安装docker4、配置阿里云镜像加速器5、docker run 执行顺序6、docker和虚拟机比较三、docker常用命令四、docker镜像分层一、什么是Docker&…