news 2026/4/15 11:31:11

【Java-JMM】Happens-before原则

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【Java-JMM】Happens-before原则

心理解:如果操作 A happens-before 操作 B,那么 A 的执行结果对 B 可见。这个原则主要解决了 Java 并发编程中的两个关键问题:

可见性问题:由 CPU 缓存引起

有序性问题:由编译器优化和指令重排引起

二、Happens-before 的具体规则#

1. 程序顺序性规则#

在单线程中,按照程序代码顺序,前面的操作 happens-before 后面的操作。

关键点:

有依赖关系:操作间存在数据依赖时,顺序不可重排

无依赖关系:操作间无数据依赖时,可以重排序,但要保证单线程执行结果不变

int a = 1; // 操作A

int b = 2; // 操作B(与A无依赖,可重排)

int c = a + 1; // 操作C(依赖A,必须在A之后)

int d = b * 2; // 操作D(依赖B,必须在B之后)

// 可能的执行顺序:

// ✓ A → B → C → D(原始顺序)

// ✓ B → A → C → D(B与A无依赖,可交换)

// ✗ C → A → B → D(C依赖A,不能在A之前)

2. volatile 变量规则#

对 volatile 变量的写操作 happens-before 后续对该变量的读操作。

volatile int flag = 0;

// 线程A

flag = 1; // 写操作

// 线程B

if (flag == 1) { // 读操作

// 能看到线程A的写入

}

3. 传递性规则#

如果 A happens-before B,且 B happens-before C,那么 A happens-before C。

4. 锁规则(Monitor Lock Rule)#

对一个锁的解锁操作 happens-before 后续对这个锁的加锁操作。

synchronized (lock) {

// 临界区代码

} // 解锁

// 其他线程

synchronized (lock) { // 加锁

// 能看到前一个线程在临界区的所有操作

}

5. 线程启动规则#

线程 A 中调用线程 B 的 start() 方法之前的所有操作,happens-before 线程 B 中的任意操作。

6. 线程终止规则#

线程 B 中的所有操作 happens-before 线程 A 中调用 B.join() 方法成功返回后的操作。

public class VisibilityDemo {

static int var = 0;

public static void main(String[] args) throws InterruptedException {

// 主线程操作

var = 10; // ① 主线程修改

Thread B = new Thread(() -> {

// 子线程B能看到①的修改(线程启动规则)

var = 66; // ② 子线程修改

});

B.start(); // 启动子线程

B.join(); // 等待子线程结束

// ③ 主线程能看到②的修改(线程终止规则)

System.out.println(var); // 输出:66

}

}

执行流程:

根据线程启动规则:主线程的 var = 10 happens-before 子线程 B 的所有操作

根据线程终止规则:子线程 B 的 var = 66 happens-before 主线程 join() 之后的操作

因此主线程最终能看到 var 的值为 66

7. final 字段规则#

在构造函数中对 final 字段的写入,happens-before 其他线程对该对象的 final 字段的读取。

public class FinalExample {

private final int value;

public FinalExample(int value) {

this.value = value; // 构造函数中的写入

}

// 其他线程读取时,保证能看到构造函数中的赋值

public int getValue() {

return value;

}

}

三、总结#

Happens-before 原则是 Java 并发编程的基石,它通过定义操作间的可见性关系,让开发者能够在不了解底层硬件细节的情况下,编写正确的并发程序。掌握这些规则,是写出线程安全代码的关键。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/14 3:25:57

多智能体协同系统

多智能体协同系统的核心概念 多智能体协同系统(Multi-Agent Systems, MAS)通过多个自主智能体的交互实现复杂任务,广泛应用于机器人协作、自动驾驶、游戏AI等领域。核心特性包括分布式决策、通信协议、任务分配与冲突解决。典型应用案例 1. 无…

作者头像 李华
网站建设 2026/4/15 3:51:52

多角度关于人的本质的论述,你怎么思考?

第六章:多角度关于人的本质的论述人的本质,人和动物的区别是什么,此文可以参考。这个问题很深奥,历来人类试图回答。比如中国古代对于人,有善恶之分,但这显然不具有说服力。以下是马克思哲学关于人本质的思…

作者头像 李华
网站建设 2026/4/14 5:21:25

Flutter 实现一个容器内部元素可平移、缩放和旋转等功能(六)

Flutter 实现一个容器内部元素可平移、缩放和旋转等功能(六) Flutter: 3.35.6 前面有人提到在元素内部的那块判断怎么那么写的,看来对知识渴望的小伙伴还是有,这样挺好的。不至于说牢记部分知识,只需要大致了解一下有…

作者头像 李华
网站建设 2026/4/14 10:30:03

python作业4

a 56 b -18# 1. 按位与(&):对应位都为1则为1,否则为0 # 56: 00111000 # -18补码: 11101110 # 按位与: 00101000 → 十进制40 bit_and a & b print(f"按位与(&): {a} & {b} {bit_and}")# 2. 按位或(|):对应位有…

作者头像 李华
网站建设 2026/4/13 10:20:46

今天教大家免费使用先进的AI大模型,非常详细收藏这一篇就够了

为什么要使用ai模型? 用好ai可以解决你想做的事情比如数据录入、数据整理、数据分析、数据报告等等问题。只要你想好规则,他都可以给你生成,而且你要担心数据泄露问题,完全可以让他给你生成一个离线的app或者exe程序或者前端程序&…

作者头像 李华
网站建设 2026/4/13 7:32:16

边缘AI与端云协同架构

边缘AI与端云协同架构概述 边缘AI将人工智能模型部署在边缘设备(如手机、传感器、嵌入式设备)上,实现本地实时处理;端云协同通过边缘与云计算的协作,平衡计算负载、隐私与延迟。典型应用包括智能家居、工业检测、自动驾…

作者头像 李华