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

【c++】内存序 和 内存一致性模型

c++ 11 中为了支持并发,定义了内存序和内存一致性模型。这个概念听起来非常高深,好像是在多线程编程领域浸淫多年之后的神级程序员才能搞明白,并用明白的东西。

本文尝试用最简单的方式说清楚这个概念。因为这个概念真的超级简单,大家真的被这个概念欺骗了。

我用一句话就能说清楚。内存一致性模型的本质是防止 编译器和CPU过度优化。意思是告诉 编译器和 CPU 在什么情况下,不要做乱序处理。

编译器和CPU优化

举个例子:

std::atomic<bool> valVailable(false); 
auto imptValue = computeImportantValue();   //计算值
valAvailable = true;                        //告诉另一个任务,值可用了

假设线程 A 在执行上面的代码,线程B 在轮询变量 valAvailable。当 valAvailable 变为 true 时,使用 imptValue 变量做自己的事情。

这种情况下,imptValue 变量在 valAvailable 之前被赋值就很重要了。从代码上看,这不是明摆着么?

但事实上,如果 valAvailable 不是原子变量,所有编译器看到的是给相互独立的变量的一对赋值操作。通常来说,编译器会被允许重排这对没有关联的操作。变成

valAvailable = true;                        //告诉另一个任务,值可用了
auto imptValue = computeImportantValue();   //计算值

最终造成错误。
即使编译器没有重排顺序,底层硬件也可能重排(乱序执行)(或者可能使它看起来运行在其他核心上),因为有时这样代码执行更快。

原子变量与内存序

内存序往往和原子变量一起使用。普通变量意味着内存序不重要。因此把普通变量用在并发场景是绝对的错误。

上面的代码成立还有一个保证,就是c++对原子变量的默认内存序要求是:std::memory_order_seq_cst,意味着 全局顺序一致性。在本次操作之前的所有读写操作,不允许重排到后面,后面所有的读写也不允许重排在我前面。

此外,c++ 还有另外 5 种内存序

  • std::memory_order_acq_rel:同时具有 acquire 和 release 语义
    对于当前操作:

    • 读操作:具有 acquire 语义,确保后续操作不会重排到它之前。

    • 写操作:具有 release 语义,确保前面的操作不会重排到它之后。

  • std::memory_order_release:禁止前面写操作重排到它之后,确保本线程的修改对其他线程的获取操作可见。

  • std::memory_order_acquire:禁止后续读操作重排到它之前,确保能观察到其他线程的释放操作。

  • std::memory_order_consume:保证数据依赖顺序(已不推荐使用)。

  • std::memory_order_relaxed:仅保证原子性,无顺序约束,就是说编译器和 cpu 可以随便重排,只要保证本次操作是原子的就行。

具体该怎么用,我就不讲了。知道概念就行

  1. 内存序是为了防止编译器和cpu乱序执行
  2. 内存序只结合原子变量在多线程编程下使用,普通变量没必要整这个概念。
  3. c++ atomic 变量 默认内存序设置 是禁止原子操作前后的一切重排序的,所以不懂内存序也可以随便用 atomic 而不出错。

相关文章:

  • HAL库实现SPI读写FLASH(W25Q32)
  • 组合Composition(has-a)
  • python日期
  • [Mysql]创建数据库基础
  • Rocky Linux 9.x 基于 kubeadm部署k8s 1.32
  • 【H2O2 | 软件开发】前端深拷贝的实现
  • 基于javaweb的SpringBoot公司日常考勤系统设计与实现(源码+文档+部署讲解)
  • CMake基础之-控制流
  • 【备赛】遇到的小问题-1
  • 施磊老师高级c++(一)
  • 8. Merge Sorted Array
  • 【C++基础】CMake项目
  • 使用 Spring 的 FactoryBean 创建和获取 Bean 对象
  • 将 VOC 格式 XML 转换为 YOLO 格式 TXT
  • 数据结构 -- 二叉树的存储结构
  • 使用python seaborn创建配对图:从核心概念到实战案例
  • 跨系统投屏:Realme手机(远程)投屏到Linux系统的简单方法
  • winx64 安装对应版本火狐浏览器驱动
  • 图生生AI修图,用“画面扩充“重构创作想象
  • 基于FPGA的3U机箱温度采集板PT100,应用于轨道交通/电力储能等
  • 西藏日喀则市拉孜县发生5.5级地震,震感明显部分人被晃醒
  • 第一集丨《亲爱的仇敌》和《姜颂》,都有耐人寻味的“她”
  • 成都锦江区一在建工地起火,致2人遇难1人受伤
  • 河南省省长王凯在郑州调研促消费工作,走访蜜雪冰城总部
  • 澎湃研究所“营商环境研究伙伴计划”启动
  • 定位真核生物起源于约27.2亿年前,华东师大团队在《自然》发文