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

理解原子变量之四:可见性、memory_order_acquire与 memory_order_release

在理解原子变量之二:从volatile到内存序-进一步的认识-CSDN博客里的实例3里面,我们看到,如果在一个线程里修改原子变量done,另一个线程读取done时,就会看到这个变化。这就是原子变量的可见性。其实这个认识是不全面的。在std::memory_order - cppreference.com

里面这样描述 可见性:

9cb870d920e6408f9bdd3559e3815f16.png

 最重要的是红线标出的这句话。它的意思是,假如如下两个条件同时满足:

1 a线程对某原子变量进行了写入操作,且使用了内存序memory_order_release,

2 b线程对同一个原子变量进行读操作,且使用了内存序memory_order_acquire

则b线程读取该原子变量后,不仅可以看到该原子变量的更新值(也就是a线程写入的值),还能看到a线程对该原子变量进行写入之前的所有“其他的写操作”的效果,这些"其他的写操作",不论是针对该原子变量的,还是针对其他变量的,也不论这些变量是不是原子变量,也不论使用何种内存序。

看下面示例(来自std::memory_order - cppreference.com)

#include <atomic>
#include <cassert>
#include <string>
#include <thread>std::atomic<std::string*> ptr;
int data;void producer()
{std::string* p = new std::string("Hello");data = 42;//a线程写入原子变量前的写操作,data只是一个普通变量ptr.store(p, std::memory_order_release);//线程写入原子变量
}void consumer()
{std::string* p2;while (!(p2 = ptr.load(std::memory_order_acquire)))//b线程读取原子变量;assert(*p2 == "Hello"); // 绝无问题assert(data == 42); // 绝无问题,
//因为data=42这句话,发生在a线程写入原子变量之前,
//所以在b线程读取原子变量后,data的更新值对b线程是可见的
}int main()
{std::thread t1(producer);//这就是线程astd::thread t2(consumer);//这就是线程bt1.join(); t2.join();
}

尽管data是一个普通变量,但是data=42这个操作发生在ptr.store之前(且采用内存序release),所以consumer函数读取ptr.load之后(且采用内存序acquire),也可以看到data的最新值42

但是注意,假如还有第三个线程,且第三个线程不操作原子变量ptr,或者虽然操作ptr,但是使用比较弱的内存序,如前一篇所说的relaxed,则release之前发生的写入操作的效果,不一定能像使用acquire那样被第三个线程观察到。

在本系列第二篇的结尾曾经提到,原子操作包含三个功能:原子性、可见性和限制指令重排。我们已经在本系列第三篇交代了原子性--它是所有6个内存序都要满足的要求,acquire和release当然不能例外;本篇已经介绍了acquire和release的可见性;那么下一篇就要介绍acquire和release如何限制指令重排。

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

相关文章:

  • requests的session
  • MySQL面试题补
  • Windows系统编程 - 进程遍历
  • 从零开始创建一个 Vue3+Vite+TS+Eslint项目
  • Python操作neo4j库py2neo使用(一)
  • Oracle - 多区间按权重取值逻辑 ,分时区-多层级-取配置方案(二)
  • leetcode刷题记录(四十二)——101. 对称二叉树
  • AutoDL安装docker问题
  • C++头文件大全(要是还有请帮忙)
  • 深度学习实战人脸识别
  • oracle排查长时间没提交的事务造成的阻塞案例
  • React第七节 组件三大属性之 refs 的用法注意事项
  • 工程企业需要什么样的物资管理系统?为什么需要物资管理系统?
  • 基于网页的大语言模型聊天机器人
  • 深入理解索引(一)
  • 动态规划子数组系列一>最长湍流子数组
  • MATLAB矩阵元素的修改及删除
  • 对 TypeScript 中函数如何更好的理解及使用?与 JavaScript 函数有哪些区别?
  • ubuntu搭建k8s环境详细教程
  • ubuntu安装Eclipse
  • C#里怎么样使用线程暂停?
  • 畅听FM 3.0.0 | 很有果味的电台软件,超多FM电台,支持播放本地音乐
  • 力扣面试经典 150(上)
  • 鸿蒙开发-音视频
  • 第一个autogen与docker项目
  • 第三十四篇 MobileNetV1、V2、V3模型解析
  • Python学习——字符串操作方法
  • 力扣—15.三数之和
  • 容器安全检测和渗透测试工具
  • sqlite3自动删除数据的两种设置方式记录