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

jmm,`as - if - serial` 与 `happens - before` 原则

在Java并发编程中,as - if - serialhappens - before 原则是确保程序在多线程环境下正确执行的重要规则,下面为你详细讲解:

as - if - serial原则

  1. 定义as - if - serial 原则是指,不管编译器和处理器如何优化,单线程程序的执行结果都不能被改变。编译器、处理器会遵守这个原则对单线程程序的执行顺序进行优化,保证最终执行结果和代码按照顺序执行的结果一致。
  2. 目的:它的主要目的是在不改变程序执行结果的前提下,尽可能地提高单线程程序的执行效率。因为在单线程环境下,程序员无需担心多线程带来的并发问题,编译器和处理器可以通过重排序等优化手段,让程序运行得更快。
  3. 示例
int a = 1;
int b = 2;
int c = a + b;

在这个简单的单线程代码片段中,编译器和处理器可能会对指令进行重排序。例如,它们可能先执行 int b = 2;,再执行 int a = 1;,最后执行 int c = a + b;。但无论如何重排序,最终 c 的值都会是 3,不会影响程序的执行结果,这就是 as - if - serial 原则的体现。

happens - before原则

  1. 定义happens - before 原则是Java内存模型(JMM)中定义的一种偏序关系,用于描述两个操作之间的内存可见性。如果操作A happens - before 操作B,那么操作A的执行结果对操作B是可见的,并且操作A的执行顺序在操作B之前。这里的“可见”不仅包括数据的可见性,还包括对内存操作顺序的保证。
  2. 具体规则
    • 程序顺序规则:在一个线程内,按照代码顺序,书写在前面的操作 happens - before 书写在后面的操作。例如:
int a = 1; // 操作A
int b = a + 1; // 操作B

这里操作A happens - before 操作B,因为在同一个线程内,代码顺序决定了执行顺序和可见性。
- 监视器锁规则:对一个锁的解锁操作 happens - before 后续对这个锁的加锁操作。例如:

synchronized (this) {// 临界区1,锁的加锁操作int a = 1;
} // 锁的解锁操作synchronized (this) {// 锁的加锁操作int b = a + 1; // 这里能看到临界区1中a的赋值结果
}

第一个 synchronized 块的解锁操作 happens - before 第二个 synchronized 块的加锁操作,所以第二个 synchronized 块能看到第一个 synchronized 块中对 a 的赋值。
- volatile变量规则:对一个 volatile 变量的写操作 happens - before 后续对这个 volatile 变量的读操作。例如:

volatile int a;Thread thread1 = new Thread(() -> {a = 1; // 对volatile变量a的写操作
});Thread thread2 = new Thread(() -> {int b = a; // 对volatile变量a的读操作,能看到thread1中对a的赋值
});

由于 avolatile 变量,所以 thread1 中对 a 的写操作 happens - before thread2 中对 a 的读操作。
- 线程启动规则Thread 对象的 start() 方法 happens - before 此线程的每一个动作。例如:

Thread thread = new Thread(() -> {int a = 1;
});
thread.start(); // start()方法happens - before线程内部的操作

start() 方法的调用 happens - before 线程内部对 a 的赋值操作,保证线程启动后能正确执行内部代码。
- 线程终止规则:线程中的所有操作都 happens - before 对此线程的终止检测。例如:

Thread thread = new Thread(() -> {int a = 1;
});
thread.start();
thread.join(); // 等待线程终止,线程内部所有操作happens - before这里

thread.join() 能保证在其之前线程内部的所有操作都已完成。
- 线程中断规则:对线程 interrupt() 方法的调用 happens - before 被中断线程的代码检测到中断事件的发生。例如:

Thread thread = new Thread(() -> {while (!Thread.currentThread().isInterrupted()) {// 循环}
});
thread.start();
thread.interrupt(); // interrupt()方法调用happens - before线程内部对中断的检测

interrupt() 方法的调用 happens - before 线程内部对中断状态的检测。
- 对象终结规则:一个对象的初始化完成(构造函数执行结束) happens - before 它的 finalize() 方法的开始。
3. 作用happens - before 原则为多线程编程提供了一种内存可见性的保障机制。通过这些规则,程序员可以判断在多线程环境下,一个操作的结果是否对另一个操作可见,从而避免数据竞争和其他并发问题。同时,它也为编译器和处理器的优化提供了一定的限制,即优化不能违反 happens - before 原则,以确保多线程程序的正确性。

as - if - serial 原则主要针对单线程程序的优化,而 happens - before 原则则重点解决多线程环境下的内存可见性和操作顺序问题,两者共同保证了Java程序在不同执行环境下的正确性和高效性。

http://www.lryc.cn/news/581480.html

相关文章:

  • 【一起来学AI大模型】算法核心:数组/哈希表/树/排序/动态规划(LeetCode精练)
  • OpenSearch 向量搜索与Qwen3-Embedding 集成示例
  • @Data、@AllArgsConstructor、@NoArgsConstructor不生效。lombok不起作用怎么解决?
  • Web前端开发-Vue
  • 多人协同开发时Git使用命令
  • 锁和事务的关系
  • 深入探索开源爬虫MediaCrawler,从入门到掌握多平台数据收集
  • HarmonyOS学习6 --- 数据存储
  • 9. 【Vue实战--孢子记账--Web 版开发】-- 账户账本管理(二)
  • MySQL CDC与Kafka整合指南:构建实时数据管道的完整方案
  • 1.线性神经网络--线性回归
  • 华为云 银河麒麟 vscode远程连接
  • 前端开发问题:SyntaxError: “undefined“ is not valid JSON
  • Flutter 每日翻译之 Widget
  • Vue+Openlayers加载OSM、加载天地图
  • java学习——guava并发编程练习
  • 【Guava】1.0.设计虚拟机的方向
  • 第一个Flink 程序:词频统计 WordCount(流处理)
  • LeetCode--41.缺失的第一个正数
  • 《Redis》缓存与分布式锁
  • AGV选型指南:AGV智能搬运车智能问答系统助力从技术参数到供应商选择的完整方案
  • Flutter 项目开启 UI 层级虚线(UI Guides)
  • 深度学习篇---简单果实分类网络
  • JAVA 项目找不到符号
  • 零依赖Web数据管理系统:midb轻松管理
  • Node.js EventEmitter 深入解析
  • 数据挖掘:从理论到实践的深度探索
  • C++学习之STL学习:list的模拟实现
  • DTW模版匹配:弹性对齐的时间序列相似度度量算法
  • 处理GET请求:在Web开发中如何处理GET请求