0

0

Java内存屏障(Memory Barrier)在Java中如何工作_底层指令与并发安全

P粉602998670

P粉602998670

发布时间:2026-03-01 13:08:03

|

523人浏览过

|

来源于php中文网

原创

volatile读插入loadload+loadstore屏障,写插入storestore+storeload屏障;它仅保障单次读写可见性与有序性,不保证复合操作原子性。

java内存屏障(memory barrier)在java中如何工作_底层指令与并发安全

Java里的volatile到底插了哪道内存屏障?

它不是“加个关键字就线程安全”,而是编译器和CPU执行时被强制插入特定屏障指令。JVM对volatile字段的读写,会在字节码层面生成getfield/putfield,再由JIT在生成机器码时按JSR-133规范插入屏障——读操作前加LoadLoad+LoadStore,写操作后加StoreStore+LoadStore。

常见错误是以为volatile能保证复合操作原子性,比如i++(读-改-写三步),其实只对单次读或写生效。它不阻止重排序,只是让重排序遵守“禁止越过屏障”的边界。

  • volatile写:禁止该写与前面任何读/写重排序(StoreStore);也禁止该写与后面任意读重排序(StoreLoad)
  • volatile读:禁止该读与后面任何读/写重排序(LoadLoad + LoadStore)
  • 注意:x86平台没有真正的StoreLoad屏障指令(靠mfencelock addl模拟),但JVM仍会生成,因为语义必须跨架构一致

JMM如何把Java代码映射成CPU级屏障指令?

Java内存模型(JMM)是抽象契约,不规定具体指令,但HotSpot在不同CPU上会选最轻量等效实现。比如在x86上,volatile读几乎不需额外指令(靠缓存一致性协议MESI保障可见性),而volatile写需lock addl $0, (rsp)这类带锁前缀的空操作来触发StoreStore+StoreLoad语义。

ARM或RISC-V就没这么幸运:它们弱内存模型,每次volatile读都要ldar(load-acquire),写都要stlr(store-release),否则可能看到乱序结果。

立即学习Java免费学习笔记(深入)”;

飞书知识问答
飞书知识问答

飞书平台推出的AI知识库管理和智能搜索工具

下载
  • x86下volatile写开销极小,常被误认为“没代价”,但这是架构红利,不是语言特性
  • ARM64上volatile读写都带明确acquire/release语义,性能影响比x86明显
  • 不要依赖Unsafe.loadFence()/Unsafe.storeFence()去手动补屏障——它们是内部API,JDK 9+已标记为@Deprecated(forRemoval = true)

为什么synchronized块里不用volatile也能保证可见性?

因为synchronized退出时隐式插入了StoreStore+StoreLoad屏障(对应monitorexit字节码),进入时插入LoadLoad+LoadStore(对应monitorenter)。这和volatile写+读的组合效果一致,但作用范围是整个临界区。

典型误区是给synchronized里的变量再加volatile修饰——既无必要,还可能误导后续维护者以为该字段需独立同步。

  • 临界区外的变量修改,若未通过同步机制发布,即使在块内赋值,其他线程仍可能看不到最新值
  • synchronized提供的是“原子性+可见性+有序性”三重保障,volatile只保后两者
  • 如果只是想防止指令重排(如双重检查锁里的instance赋值),volatilesynchronized更轻量,但别指望它锁住逻辑

使用VarHandle替代volatile时屏障行为有啥变化?

JDK 9引入的VarHandlegetVolatile/setVolatile方法提供更细粒度控制,底层仍走相同屏障逻辑,但绕过了字段访问检查,且支持数组元素、堆外内存等场景。

容易踩的坑是直接用get/set(非volatile版本)——它们不带任何屏障,等价于普通变量访问,哪怕目标字段本身是volatile修饰的也没用。

  • VarHandle.getVolatile() ≡ 字段上volatile读,但可动态绑定到任意内存位置
  • VarHandle.compareAndSet() 在x86上编译为cmpxchg,自带完整屏障;在ARM上则是cas指令,同样满足acquire-release语义
  • 别用VarHandle.getOpaque()做并发通信——它只禁止单向重排序(如禁止该读与前面读重排),不保证可见性

真正难的是理解“屏障不是魔法,它只约束指令顺序,不解决数据竞争”。比如两个线程同时对同一volatile int执行incrementAndGet(),结果仍可能错——因为CAS循环里包含非原子的读+计算+写三步,屏障只管每一步的边界,不管中间逻辑是否串行。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

890

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

595

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

294

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

210

2025.08.29

c++中volatile关键字的作用
c++中volatile关键字的作用

本专题整合了c++中volatile关键字的相关内容,阅读专题下面的文章了解更多详细内容。

72

2025.10.23

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

429

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

599

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

723

2023.08.10

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

24

2026.02.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4万人学习

C# 教程
C# 教程

共94课时 | 10.5万人学习

Java 教程
Java 教程

共578课时 | 74.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号