happens-before 关系

2、happens-before 关系

在 Java 中,volatile 关键字用于变量的修饰,它确保对该变量的所有读写操作都是直接从主内存中进行的,而不是从线程的本地缓存

中读取。volatile 关键字可以保证某些类型的内存可见性,并在一定程度上防止指令重排序。具体来说,volatile 可以建立一种特殊

的 happens-before 关系,确保多线程程序的正确性和一致性。

happens-before 关系是 Java 内存模型(JMM)中的一种重要概念,用于定义线程之间操作的顺序性。简单来说,如果操作 A happens-

before 操作 B,那么操作 A 的结果对操作 B 是可见的,并且操作 A 的顺序在操作 B 之前。

对于 volatile 变量,有以下几种 happens-before 关系:

  1. volatile 变量的写操作 happens-before 随后的读操作
    • 如果线程 A 对一个 volatile 变量进行写操作,然后线程 B 对这个 volatile 变量进行读操作,那么在 A 线程中对这volatile 变量的写操作 happens-before B 线程中的读操作。这意味着线程 B 将看到线程 A 写入的最新值。
  2. volatile 变量的写操作会禁止写之前的所有操作被重排序到写操作之后
    • 在对 volatile 变量进行写操作之前的所有操作,在内存模型上会被“刷回”主内存。即对 volatile 变量的写操作之前的所有普通变量的操作都将在写操作之前完成,并且在写操作之前的所有操作对后续的任何线程都是可见的。
  3. volatile 变量的读操作会禁止读之后的所有操作被重排序到读操作之前
    • 在对 volatile 变量进行读操作之后的所有操作,在内存模型上会从主内存读取最新值。即对 volatile 变量的读操作之后的所有普通变量的操作都将在读操作之后完成,并且在读操作之后的所有操作将看到写操作之后的最新结果。
示例代码

以下是一个简单的代码示例,展示了 volatile 的 happens-before 关系:

public class VolatileHappensBeforeExample {
//    private volatile boolean flag = false;private boolean flag = false;private int counter = 0;public void writer() {counter = 1;        // 普通写操作flag = true;        // volatile 写操作}public void reader() {if (flag) {         // volatile 读操作System.out.println(counter);  // 普通读操作}}public static void main(String[] args) {VolatileHappensBeforeExample example = new VolatileHappensBeforeExample();for (int i = 0; i < 10; i++) {// 创建写线程Thread writerThread = new Thread(() -> {example.writer();});// 创建读线程Thread readerThread = new Thread(() -> {example.reader();});writerThread.start();readerThread.start();}}
}

在这个示例中:

  • writer() 方法中,对 flag 的写操作 happens-before 随后的 reader() 方法中对 flag 的读操作。
  • 因此,如果 reader() 方法检测到 flagtrue,则它必然会看到 counter 的值为 1(即 writer() 方法中的写操作已发生)。

这种 happens-before 关系确保了多线程环境中的变量更新对于其他线程是可见的,从而保证了线程之间的正确通信。

上面这么啰里巴嗦地讲,这也太抽象了,即使去掉 volatile 修饰其实也不一定会出现打印不出来1的情况,必须整个程序验证一下。

验证 happens-before 关系

验证不使用 volatile 关键字会导致错误,可以通过编写一个多线程测试程序,观察在不同线程之间的共享变量是否会出现不可见性问题。具体来说,可以通过运行代码并检测在某些情况下是否会出现预期之外的结果,例如永远不会打印出预期的值

要验证不使用 volatile 关键字会导致错误,可以通过编写一个多线程测试程序,观察在不同线程之间的共享变量是否会出现不可见性问题。具体来说,可以通过运行代码并检测在某些情况下是否会出现预期之外的结果,例如永远不会打印出预期的值。

验证代码

以下是一个示例代码,通过多个线程的交互来验证如果不使用 volatile 关键字会出现的问题:

public class VolatileHappensBeforeExample {
//    private boolean flag = false;  // 没有使用 volatileprivate volatile boolean flag = false;  // 使用 volatileprivate int counter = 0;public void writer() {counter = 1;        // 普通写操作flag = true;        // 普通写操作System.out.println(Thread.currentThread().getName() + " set flag to true");}public void reader() {while (!flag) {// Busy-wait loop, waiting for flag to become true}System.out.println(Thread.currentThread().getName() + " sees flag is true and counter is " + counter);}public static void main(String[] args) {VolatileHappensBeforeExample example = new VolatileHappensBeforeExample();// 创建写线程Thread writerThread = new Thread(() -> {try {Thread.sleep(100);  // 确保 reader 线程先启动} catch (InterruptedException e) {Thread.currentThread().interrupt();}example.writer();}, "WriterThread");// 创建读线程Thread readerThread = new Thread(() -> {example.reader();}, "ReaderThread");readerThread.start();writerThread.start();try {readerThread.join();writerThread.join();} catch (InterruptedException e) {Thread.currentThread().interrupt();}}
}

代码解释

  1. writer() 方法
    • 设置 counter 为 1。
    • 设置 flag 为 true。
    • 打印当前线程名称及其操作。
  2. reader() 方法
    • 使用一个 busy-wait 循环等待 flag 变为 true。
    • flag 为 true 时,打印当前线程名称及其看到的 counter 值。
  3. main 方法
    • 创建并启动 writerThreadreaderThread
    • 使用 Thread.sleep(100) 确保 readerThread 先启动。

可能的结果

运行上述代码多次,可能会看到以下结果:

  • 有时,程序会如预期输出 WriterThread set flag to trueReaderThread sees flag is true and counter is 1
  • 但在某些运行中,可能会看到 WriterThread set flag to true,但 readerThread 进入 busy-wait 循环后永远不会退出。这是因为 readerThread 可能无法看到 flag 被设置为 true 的更新。
结论

如果不使用 volatile 关键字,flag 的写入更新对其他线程不可见,导致 readerThread 无法检测到 flag 的变化并一直在 busy-wait 循环中。这验证了不使用 volatile 关键字时可能出现的内存可见性问题。

通过多次运行这个程序,观察到 readerThread 不会始终成功读取到 flag 的变化,就可以确认不使用 volatile 关键字会导致多线程程序中的错误。这种错误在 volatile 关键字存在时不会发生,因为 volatile 能确保内存可见性和建立正确的 happens-before 关系。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/847807.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

createAsyncThunk完整用法介绍

createAsyncThunk 是 Redux Toolkit 库中的一个功能&#xff0c;它用于创建处理异步逻辑的 thunk action creator。Redux Toolkit 是一个官方推荐的库&#xff0c;用于简化 Redux 开发过程&#xff0c;特别是处理常见的 Redux 模式&#xff0c;如异步数据流。createAsyncThunk …

ubuntu安装或删除cmake

如果你是第一次安装cmake&#xff0c;直接敲如下命令就可以了&#xff1a; sudo apt update sudo apt-get install cmake 如果发现之前安装的cmake不能被损坏或者不能用了&#xff0c;就要把之前的cmake重新安装&#xff0c;那么我们就要先卸载之前的cmake&#xff0c;敲如下…

机器视觉——找到物块中心点

首先先介绍一下我用的是HALCON中的HDevelop软件。 大家下载好软件后可以测试一下&#xff1a; 在程序编辑器窗口中输入下面指令&#xff1a; read_image(Image,monkey) 那么如果出现这样的图片&#xff0c;说明是没有问题的 那么本次编程采用的是下面这张图片 我们要达到的…

CompletableFuture的底层他是如何实现的

底层原理 CompletableFuture是Java8中引入的新特性,像我们平常遇到的, 某一个计算过程依赖于另外一个计算过程的结构,这些结构就类似于一种链表的形式来体现,每一个阶段都代表着一种异步的操作, 然后这些阶段又是相互依赖的, 基于一种链式的操作. 官方点来解释就是 每个 Comple…

【论文复现|智能算法改进】基于改进麻雀算法的无线传感器网络覆盖优化研究

目录 1.算法原理2.改进点3.结果展示4.参考文献5.代码获取 1.算法原理 【智能算法】麻雀搜索算法&#xff08;SSA&#xff09;原理及实现 WSN数学模型 2.改进点 基于Sobol序列和ICMIC混沌映射的种群初始化 ICMIC是一种无线映射折叠次数的映射模型: { z n 1 sin ⁡ ( α π…

客户案例|Zilliz Cloud 助力点石科技转型 AI 智能服务商

福建点石科技网络科技有限公司成立于2010年&#xff0c;是国家高新技术企业&#xff0c;阿里云、蚂蚁金服等大厂海内外生态合作伙伴ISV。在餐饮、零售、酒店、旅游、商圈的行业定制化服务化上有深厚积累&#xff0c;在境内外做了大量标杆性软件项目&#xff0c;如东南亚RWS圣淘…

Vite + Vue 3 前端项目实战

一、项目创建 npm install -g create-vite #安装 Vite 项目的脚手架工具 # 或者使用yarn yarn global add create-vite#创建vite项目 create-vite my-vite-project二、常用Vue项目依赖安装 npm install unplugin-auto-import unplugin-vue-components[1] 安装按需自动导入组…

Python | 试卷刷题and基础笔记

1.下列转义字符中&#xff0c; 属于“回车”含义的是 \n 换行 \r 回车 2.for循环遍历字典 在Python中&#xff0c;你可以使用for循环来遍历字典的键&#xff08;keys&#xff09;、值&#xff08;values&#xff09;或者键-值对&#xff08;items&#xff09;。下面是三种遍历…

supervisord常用命令及服务配置记录

whereis supervisor主进程配置文件&#xff1a;/etc/supervisord.conf 子进程配置文件&#xff1a;/etc/supervisor/conf.d 启动supervisor服务 sudo supervisord -c /etc/supervisor/supervisord.conf关闭supervisor则执行命令 sudo supervisorctl shutdown重启supervisor则…

Postman接口测试笔记(超详细)

基于工具Poatman的接口自动化基础应用以及接口关联 一、什么是接口&#xff1f; 硬件接口&#xff1a;USB接口&#xff0c;投影仪接口&#xff0c;鼠标键盘接口 软件接口&#xff1a;称为API&#xff0c;主要使用于数据交互 软件接口分类&#xff1a; 内部接口&#xff1a…

Vue3实战笔记(61)—Vue 3 Watch进化论:解锁实时数据监听新境界

文章目录 前言基本使用总结 前言 Vue 3 中的 watch 功能相比Vue 2有了改进和扩展&#xff0c;旨在提供更灵活的数据监听方式。 基本使用 Vue 3中的watch可以用于Composition API和Options API&#xff0c;但Composition API的使用更为常见。它主要用于监听响应式数据的变化&a…

树莓派5烧系统和ssh远程实现

1、硬件说明 树莓派5 64G micro SD卡读卡器 2、烧录系统过程记录 之前写过一篇pi4B烧录Ubuntu22.04的博客&#xff0c;这篇就简单记录备份下 2.1 去ubuntu官网在树莓派上安装Ubuntu | Ubuntu下载Ubuntu 桌面 24.04 LTS 我之前已经下好了就有个(1) 2.2 用读卡器把SD卡插到…

富格林:采用安全出金操作方法

富格林指出&#xff0c;现货黄金投资是近几年来热门的投资产品&#xff0c;剧烈的行情波动会给投资者带来充分的盈利机会&#xff0c;让不少投资者都开始投身于黄金市场。不过对于新手而言&#xff0c;要想实现安全盈利出金并没有那么容易&#xff0c;需要把握一些正规的交易技…

实战:一款唯美的个人主页-home2.0-2024.6.4(测试成功)

目录 文章目录 目录实验软件前提条件效果说明1、背景2、配置1、克隆代码库2、配置并构建镜像3、部署测试方案1&#xff1a;从docker容器拷贝生成的静态文件放到网站/目录方案2&#xff1a;启动容器&#xff0c;nginx里配置反向代理&#xff08;推荐&#xff09; 4、访问 3、总结…

“物联网安全:万物互联背景下的隐私保护与数据安全策略“

在物联网&#xff08;IoT&#xff09;时代&#xff0c;随着智能设备的普及和万物互联的加速&#xff0c;隐私保护与数据安全成为了亟待解决的关键问题。以下是一些重要的隐私保护与数据安全策略&#xff0c;以确保在万物互联背景下信息的安全&#xff1a; 1. 加强设备安全&…

.NET Core 应用程序发布指南

引言 .NET Core 是一个开源、跨平台的框架&#xff0c;用于构建现代化的、高性能的应用程序。本文将介绍如何将一个 .NET Core 应用程序发布到不同的环境中&#xff0c;包括本地、云端和容器化部署。 准备工作 在开始之前&#xff0c;请确保您的开发环境已经安装了以下工具&…

MySQL数据库开发设计规范总结

MySQL数据库开发设计规范总结 概述MySQL数据库设计规范设计规范-库设计规范-表、列设计规范-索引跟索引相关的SQL优化设计规范-视图设计规范-存储过程设计规范-触发器设计规范-安全规范 数据库架构设计原则一、高可用架构选择二、扩展性三、安全性策略四、数据完整性策略五、规…

全球首款AR电脑上线,可投影100英寸屏幕

近日&#xff0c;Sightful公司推出了一款名为Spacetop G1的革命性笔记本电脑&#xff0c;将AR技术与传统笔记本电脑巧妙融合&#xff0c;打造出令人惊叹的全新办公体验。 全球首款AR电脑上线&#xff0c;可投影100英寸屏幕 不同于传统笔记本电脑依赖物理屏幕显示内容&#xff0…

异常概述

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 在程序运行过程中&#xff0c;经常会遇到各种各样的错误&#xff0c;这些错误统称为“异常”。这些异常有的是由于开发者将关键字敲错导致的&#xf…

重学java 63.IO流 字节流 ④ 文件复制

身处泥泞&#xff0c;看满山花开 —— 24.6.4 图片复制 分析 1.创建两个对象 FilelnputStream —>读取指定的文件 FileOutputStream —> 将读到的字节写到指定的位置 2.边读边写 import java.io.FileInputStream; import java.io.FileOutputStream;public class Demo…