当前位置: 首页 > 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.dtcms.com/a/268040.html

相关文章:

  • Dubbo 3.x源码(31)—Dubbo消息的编码解码
  • 容声W60以光水离子科技实现食材“主动养鲜”
  • 创客匠人深度剖析:家庭教育赛道创始人 IP 打造与知识变现的破局之道
  • 【算法刷题记录(简单题)003】统计大写字母个数(java代码实现)
  • 0704-0706上海,又聚上了
  • 【MyBatis】实现数据库的增、删、改、查
  • 深度解析命令模式:将请求封装为对象的设计智慧
  • 儿童趣味记忆配对游戏
  • LeetCode 75. 颜色分类(荷兰国旗问题)
  • 一次佳能iX6780彩色喷墨打印机报5B00维修的记录
  • 【网络协议安全】任务13:ACL访问控制列表
  • 牛客周赛Round 99(Go语言)
  • 《kubernetes》k8s实战之部署PHP/JAVA网站
  • 中级统计师-经济学基础知识-第四章 国民收入核算
  • 单片机物联网应用中的 Pogopin、串口与外围模组通信技术解析
  • Java 大视界 -- Java 大数据在智能教育在线课程学习效果影响因素分析与优化设计(334)
  • Zotero中进行文献翻译【Windows11】
  • SpiceMix enables integrative single-cell spatial modeling of cell identity 文章解读
  • 【kafka-python使用学习笔记1:Python操作Kafka之环境准备(1)】
  • 2、Connecting to Kafka
  • css模块化以及rem布局
  • linux/ubuntu日志管理--/dev/log 的本质与作用
  • arm 精准总线错误与非精准总线错误
  • C#使用Qdrant实现向量存储及检索
  • 基于ARM+FPGA的光栅尺精密位移加速度测试解决方案
  • 【精密测量】基于ARM+FPGA的多路光栅信号采集方案
  • 【PyTorch 当前版本不支持 NVIDIA GeForce RTX 5060 Ti处理办法】
  • 求医十年,病因不明,ChatGPT:你看起来有基因突变
  • 群晖(Synology)存储ext4视频文件删除的恢复方法
  • Java--指定控制台System.out.println的颜色