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

「JVM 高效并发」锁优化

为了线程间更高效的共享数据及解决竞争问题,提高程序执行效率,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 虚拟机》

相关文章:

「JVM 高效并发」锁优化

为了线程间更高效的共享数据及解决竞争问题,提高程序执行效率,JDK 6 做了大量锁优化,如适应性自旋(Adaptive Spinning)、锁消除(Lock Elimination)、锁膨胀(Lock Coarsening&#xf…...

当园区物流遇上云计算,会发生什么事情?

顺丰供应链与亚马逊云科技的强强联手,可以给物流供应链企业带来怎样的启示?物流行业的数智化趋势在国内物流行业说起顺丰,相信是无人不知无人不晓。作为数字化供应链服务解决方案提供商,顺丰供应链可以提供端到端供应链的规划、管…...

作为测试开发岗的面试官,我都是怎么选人的?

最近一段时间面试了不少人,主要是一些测试开发岗,中高级的初级的也都有;也有一些偏业务测试岗的候选人。总结出了一些方法论,或者说更多的是个人作为面试官所遵守的一套面试准则。 1.什么是面试? 面试不仅仅是你问我…...

android事件分发机制源码分析

没什么用的前言责任链设计模式流程图源码分析 没什么用的前言 事件分发机制是面试中一道必问的题目,而我的应对方式则是,在网络上找一些博客看看,然后做一些笔记,最后在面试时将我自己记住的内容说出来。这种方式本身没有太大的…...

今天,小灰37岁了!

人们常常说,35岁是互联网人的中年危机。现在,小灰已经跨过了中年危机,倒不是因为小灰财务自由了,而是因为今天是小灰37岁的生日。年轻时候,小灰总觉得30岁是一个很遥远的年龄,而现在,小灰距离40…...

基于.NET 7 + iView 的前后端分离的通用后台管理系统开源框架

更多开源项目请查看:一个专注推荐.Net开源项目的榜单 今天给大家推荐一套前后端分离通用后台管理系统开源框架。 项目简介 这是基于.Net 7 Vue.js开发的、前后端分离框架,前端UI框架采用iView,该项目只有基础功能模块,不包含具…...

新一代通信协议—— RSocket

一、简介 RSocket 是一种二进制字节流传输协议,位于 OSI 模型中的5~6层,底层可以依赖 TCP、WebSocket、Aeron 协议。最初由 Netflix 开发,支持 Reactive Streams。其开发背后的动机是用开销更少的协议取代超文本传输协议(HTTP),H…...

【编程实践】这个代码命名规范是真优雅呀!代码如诗!!(多读优秀的开源代码,多实践,你也可以一样优秀!)

目录 管理类命名 传播类命名 回调类命名 监控类命名 内存管理类命名 过滤检测类命名 结构类命名 常见设计模式命名 解析类命名 网络类命名 CRUD命名 其他 End 管理类命名 写代码,少不了对统一资源的管理,清晰的启动过程可以有效的组织代码…...

Linux->进程终止和等待

目录 1. 进程终止场景 1.1 进程退出码 1.2 进程常见退出方式 2. 进程等待 2.1 进程等待的必要性 2.2 进程等待的方式 wait()方式 waitpid()方式 options参数 status参数 1. 进程终止场景 代码运行完毕,结果正确 代码运行完毕,结果不正确 代码异…...

超店有数分享:tiktok数据分析工具推荐,助你成功出海!

现阶段的跨境电商人都纷纷入局tiktok,这是风口也是发展趋势。Tiktok的下载量已经超过了35亿,每月都有10亿用户活跃,在154国家/地区使用。Tiktok用户每天在平均花1小时左右进行浏览,打开率也很高。如今,tiktok也越来越成…...

2022 第十四届蓝桥杯模拟赛第三期(题解与标程)

第十四届蓝桥杯模拟赛第三期1. 最小的十六进制问题描述答案提交参考答案2. Excel的列问题描述答案提交参考答案3. 相等日期问题描述答案提交参考答案4. 多少种取法问题描述答案提交参考答案5. 最大连通分块问题描述答案提交参考答案6. 哪一天问题描述输入格式输出格式样例输入样…...

「TCG 规范解读」PC 平台相关规范(1)

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

HNU工训中心:直流电路测量分析实验报告

工训中心的牛马实验 实验目的 1.熟悉直流电路的测量和分析方法。 2.熟悉直流电源、电压表、电流表的使用法及其特性。 实验仪器和器材 1.实验仪器 直流稳压电源型号:IT6302 台式多用表型号:UT805A 2.实验(箱)器材 电路实验箱 元器件:电阻…...

tensorflow2.4--1.框架介绍

前言 虽然1.x版本tensorflow有很多项目都基于此构建,然而随着2.x版本的推出,很多架构已经发生了改变,代码发生了改变,同时很多模组已经废弃不用或者更新,tensorflow1.x已经不能再兼容最新的项目,与时俱进是必要的,因此…...

c++11 关键字 final 使用

写在最前。。。 请支持原创~~ 1. 功能 用以指定一个 virtual function 不能被派生类重写;或者指定一个 class 不能被继承;2. 语法 对于类中成员函数有两种情况: 只声明时,final 紧跟参数的右括号,如果是纯虚函数&a…...

力扣(LeetCode)426. 将二叉搜索树转化为排序的双向链表(2023.02.28)

将一个 二叉搜索树 就地转化为一个 已排序的双向循环链表 。 对于双向循环列表,你可以将左右孩子指针作为双向循环链表的前驱和后继指针,第一个节点的前驱是最后一个节点,最后一个节点的后继是第一个节点。 特别地,我们希望可以…...

华为OD机试真题Python实现【玩牌高手】真题+解题思路+代码(20222023)

玩牌高手 题目 给定一个长度为N的整数数组,表示一个选手在N轮内选择的牌面分数, 选手基于规则选牌,请计算所有轮结束后其可以获得的最高总分数。 选择规则如下: 在每轮里选手可以选择获取该轮牌面,则其总分数加上该轮牌面分数为其新的总分数选手也可不选择本轮牌面,直接…...

“速通“ 老生常谈的HashMap [实现原理源码解读]

👳我亲爱的各位大佬们好😘😘😘 ♨️本篇文章记录的为 HashMap 实现原理&&源码解读 相关内容,适合在学Java的小白,帮助新手快速上手,也适合复习中,面试中的大佬🙉🙉&#x1f…...

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

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

mysql数据库limit的四种用法

文章目录前言一、语法二、参数说明三、常用示例-4种用法总结前言 mysql数据库中limit子句可以被用于强制select语句返回指定的记录数。limit接受一个或两个数字参数。参数必须是一个整数常量。如果给定两个参数,第一个参数指定第一个返回记录行的偏移量&#xff0c…...

CVPR 2025 MIMO: 支持视觉指代和像素grounding 的医学视觉语言模型

CVPR 2025 | MIMO:支持视觉指代和像素对齐的医学视觉语言模型 论文信息 标题:MIMO: A medical vision language model with visual referring multimodal input and pixel grounding multimodal output作者:Yanyuan Chen, Dexuan Xu, Yu Hu…...

Spring Boot 实现流式响应(兼容 2.7.x)

在实际开发中,我们可能会遇到一些流式数据处理的场景,比如接收来自上游接口的 Server-Sent Events(SSE) 或 流式 JSON 内容,并将其原样中转给前端页面或客户端。这种情况下,传统的 RestTemplate 缓存机制会…...

【Redis技术进阶之路】「原理分析系列开篇」分析客户端和服务端网络诵信交互实现(服务端执行命令请求的过程 - 初始化服务器)

服务端执行命令请求的过程 【专栏简介】【技术大纲】【专栏目标】【目标人群】1. Redis爱好者与社区成员2. 后端开发和系统架构师3. 计算机专业的本科生及研究生 初始化服务器1. 初始化服务器状态结构初始化RedisServer变量 2. 加载相关系统配置和用户配置参数定制化配置参数案…...

《用户共鸣指数(E)驱动品牌大模型种草:如何抢占大模型搜索结果情感高地》

在注意力分散、内容高度同质化的时代,情感连接已成为品牌破圈的关键通道。我们在服务大量品牌客户的过程中发现,消费者对内容的“有感”程度,正日益成为影响品牌传播效率与转化率的核心变量。在生成式AI驱动的内容生成与推荐环境中&#xff0…...

《通信之道——从微积分到 5G》读书总结

第1章 绪 论 1.1 这是一本什么样的书 通信技术,说到底就是数学。 那些最基础、最本质的部分。 1.2 什么是通信 通信 发送方 接收方 承载信息的信号 解调出其中承载的信息 信息在发送方那里被加工成信号(调制) 把信息从信号中抽取出来&am…...

多模态大语言模型arxiv论文略读(108)

CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文标题:CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文作者:Sayna Ebrahimi, Sercan O. Arik, Tejas Nama, Tomas Pfister ➡️ 研究机构: Google Cloud AI Re…...

Reasoning over Uncertain Text by Generative Large Language Models

https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829 1. 概述 文本中的不确定性在许多语境中传达,从日常对话到特定领域的文档(例如医学文档)(Heritage 2013;Landmark、Gulbrandsen 和 Svenevei…...

LINUX 69 FTP 客服管理系统 man 5 /etc/vsftpd/vsftpd.conf

FTP 客服管理系统 实现kefu123登录,不允许匿名访问,kefu只能访问/data/kefu目录,不能查看其他目录 创建账号密码 useradd kefu echo 123|passwd -stdin kefu [rootcode caozx26420]# echo 123|passwd --stdin kefu 更改用户 kefu 的密码…...

【笔记】WSL 中 Rust 安装与测试完整记录

#工作记录 WSL 中 Rust 安装与测试完整记录 1. 运行环境 系统:Ubuntu 24.04 LTS (WSL2)架构:x86_64 (GNU/Linux)Rust 版本:rustc 1.87.0 (2025-05-09)Cargo 版本:cargo 1.87.0 (2025-05-06) 2. 安装 Rust 2.1 使用 Rust 官方安…...

GitFlow 工作模式(详解)

今天再学项目的过程中遇到使用gitflow模式管理代码,因此进行学习并且发布关于gitflow的一些思考 Git与GitFlow模式 我们在写代码的时候通常会进行网上保存,无论是github还是gittee,都是一种基于git去保存代码的形式,这样保存代码…...