0

0

在Java里happens before规则的意义_Java并发顺序解析

P粉602998670

P粉602998670

发布时间:2026-03-16 16:38:03

|

584人浏览过

|

来源于php中文网

原创

happens-before 是 JVM 内存模型的顺序契约,定义写操作对读操作的可见性而非执行顺序;满足该关系才保证读线程看到最新值,常见规则包括 volatile 写→读、锁释放→获取、start→run、join 返回→后续操作、final 字段安全发布等。

在java里happens before规则的意义_java并发顺序解析

happens-before 是 JVM 内存模型的“顺序契约”,不是执行顺序保证

它不决定代码实际怎么跑,而是定义哪些写操作对哪些读操作“可见”。只要满足 happens-before 关系,JVM 就必须确保读线程能看到之前写线程的最新值;不满足时,即使代码看起来是先后写的,读线程也可能看到过期值、0、null 或任意中间态。

常见误判是把 synchronized 块的执行先后等同于 happens-before——其实只有加锁/解锁配对才构成关系,比如线程 A 释放锁后,线程 B 获取同一把锁,才建立 A 对 B 的 happens-before。

  • volatile 写 happens-before 后续对该变量的 volatile 读
  • start() 调用 happens-before 线程 run() 中的任意操作
  • 线程所有操作 happens-before 其 join() 返回
  • 同一个 monitor 的 unlock happens-before 后续对同一 monitor 的 lock

volatile 为什么能禁止重排序但不能替代锁

volatile 的语义包含两层:对变量的读写直接走主内存(不缓存到寄存器或 CPU cache),以及插入内存屏障(memory barrier)禁止编译器和处理器对它前后指令的重排序。但它不提供原子性——count++ 这种读-改-写操作在 volatile 变量上仍是竞态的。

典型陷阱是以为给 boolean 标志位加了 volatile,就能安全控制整个初始化流程。如果初始化涉及多个字段,仅靠一个 volatile flag 无法保证其他字段的写入对读线程可见(缺少 happens-before 链)。

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

  • 正确做法:把所有需同步的字段一起纳入同步块,或用 final 字段 + 构造器安全发布
  • 错误写法:flag = true 前没同步写其他字段,读线程看到 flag == true 却读到未初始化的字段值
  • volatile 适合状态标志、一次性事件通知(如 shutdown)、轻量级读多写少场景

Thread.join() 的 happens-before 效果常被低估

调用 thread.join() 并返回,意味着该线程已终止,且它执行期间的所有操作(包括对共享变量的写)都对当前线程可见。这个关系比很多人想的更强:它不只是“线程结束”,而是“它做过的所有事我都看得到”。

米粿AI
米粿AI

AI辅助动漫创作工具,专为专业漫画师和动画制作团队设计

下载

注意它只对调用方生效,不影响其他线程。如果两个线程都 join 同一个线程,它们彼此之间没有 happens-before 关系。

  • 适用场景:主线程等待 worker 线程完成计算并消费结果,无需额外同步
  • 等价于在 worker 线程末尾隐式执行了 flush,主线程 join 返回后可直接读 sharedResult
  • 不要用 thread.isAlive() 替代 join() —— 它不建立 happens-before,无法保证可见性

final 字段的 happens-before 是安全发布的底层保障

对象构造器中对 final 字段的赋值,会在构造完成时对其他线程“立即可见”——前提是该对象本身是安全发布的(如通过 volatile 引用、static final、锁内发布等)。JVM 为此在构造器结尾插入 StoreStore 屏障,防止 final 字段写被重排序到构造器外。

一旦绕过安全发布(比如把 this 引用逸出到构造器中),final 的保障就失效,其他线程可能看到 partially initialized 对象。

  • 禁止在构造器里注册监听器、启动线程、放入全局 map——这些都会导致 this 逸出
  • 静态工厂方法 + private 构造器 + final 字段是推荐的安全发布模式
  • final 只保护字段引用本身不可变,不保护其指向对象的内部状态(如 final List 的元素仍可被修改)

真正难的是把 happens-before 链串起来:一个 volatile 写、一次锁释放、一个 join 返回……这些点要连成完整路径,才能让远端读取看到预期结果。漏掉任意一环,看似合理的并发逻辑就会在高并发或特定 CPU 架构下崩塌。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
java中boolean的用法
java中boolean的用法

在Java中,boolean是一种基本数据类型,它只有两个可能的值:true和false。boolean类型经常用于条件测试,比如进行比较或者检查某个条件是否满足。想了解更多java中boolean的相关内容,可以阅读本专题下面的文章。

367

2023.11.13

java boolean类型
java boolean类型

本专题整合了java中boolean类型相关教程,阅读专题下面的文章了解更多详细内容。

42

2025.11.30

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

254

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

1132

2024.03.01

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

203

2023.11.20

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

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

77

2025.10.23

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

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

786

2023.08.10

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

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

786

2023.08.10

minimax入口地址汇总
minimax入口地址汇总

本专题整合了minimax相关入口合集,阅读专题下面的文章了解更多详细地址。

3

2026.03.16

热门下载

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

精品课程

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

共23课时 | 4.5万人学习

C# 教程
C# 教程

共94课时 | 11.5万人学习

Java 教程
Java 教程

共578课时 | 83.2万人学习

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

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