0

0

C++中的内存屏障(Memory Barrier)有什么用?(防止指令重排序)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-12 13:15:12

|

821人浏览过

|

来源于php中文网

原创

内存屏障解决多线程中因编译器/CPU重排序导致的非原子变量可见性与顺序问题;memory_order_release确保其前所有内存访问不被重排到其后,memory_order_acquire确保其后所有内存访问不被重排到其前,二者配对建立happens-before关系。

c++中的内存屏障(memory barrier)有什么用?(防止指令重排序)

内存屏障解决什么问题?

在多线程 C++ 程序中,std::atomic 变量的读写本身不保证其他非原子变量的可见顺序。编译器和 CPU 都可能对指令做重排序——比如把某个非原子变量的赋值提前到原子操作之前,或延后到之后。这会导致其他线程看到“逻辑上不可能”的状态,比如看到指针已更新但指向的对象尚未初始化完毕。

memory_order_acquire 和 memory_order_release 怎么用?

它们是隐式带内存屏障的原子操作语义,比手动插入 std::atomic_thread_fence 更常用、更安全。

  • memory_order_release 用于写操作:确保该原子写之前的**所有内存访问**(包括非原子读写)不会被重排到它之后
  • memory_order_acquire 用于读操作:确保该原子读之后的**所有内存访问**不会被重排到它之前
  • 二者配对使用才能建立同步关系(happens-before),单用没意义
std::atomic ready{false};
int data = 0;

// 线程 A
data = 42;                          // 非原子写
ready.store(true, std::memory_order_release);  // 释放操作

// 线程 B
while (!ready.load(std::memory_order_acquire)) { }  // 获取操作
std::cout << data << "\n";  // 此时一定能读到 42

什么时候必须用 std::atomic_thread_fence?

当需要在**没有原子读写参与的位置**插入屏障,或者需要跨多个原子操作统一约束顺序时。例如:两个独立的原子变量之间建立顺序,或配合 relaxed 原子操作做精细控制。

  • std::atomic_thread_fence(std::memory_order_acquire) 等价于 acquire 读的屏障部分,但不读任何值
  • std::atomic_thread_fence(std::memory_order_release) 等价于 release 写的屏障部分,但不写任何值
  • 注意:fence 不作用于任何特定原子对象,影响的是当前线程所有内存访问
std::atomic flag{0};
int payload = 0;

// 线程 A
payload = 100;
std::atomic_thread_fence(std::memory_order_release);
flag.store(1, std::memory_order_relaxed);

// 线程 B
while (flag.load(std::memory_order_relaxed) == 0) { }
std::atomic_thread_fence(std::memory_order_acquire);
std::cout << payload << "\n";  // 能正确读到 100

容易忽略的关键点

内存屏障只约束**本线程内指令的重排序可见性**,不保证缓存一致性自动传播;它也不代替互斥锁——如果存在数据竞争(如两个线程同时写同一非原子变量),加屏障也解决不了未定义行为。

萝卜简历
萝卜简历

免费在线AI简历制作工具,帮助求职者轻松完成简历制作。

下载

立即学习C++免费学习笔记(深入)”;

  • 仅靠 memory_order_relaxed 的原子操作,无法建立线程间同步
  • x86/x64 上 acquire/release 几乎不生成额外指令(靠 CPU 内存模型保证),但 ARM/AArch64 必须插入 dmb 指令
  • 过度使用 memory_order_seq_cst(默认)会显著影响性能,尤其在弱序架构上

真正难的不是插哪条 fence,而是准确识别哪些变量需要同步、哪些访问必须按序——这取决于你设计的同步协议,而不是语法本身。

相关专题

更多
线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

480

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

143

2025.12.24

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

34

2026.01.14

php与html混编教程大全
php与html混编教程大全

本专题整合了php和html混编相关教程,阅读专题下面的文章了解更多详细内容。

14

2026.01.13

PHP 高性能
PHP 高性能

本专题整合了PHP高性能相关教程大全,阅读专题下面的文章了解更多详细内容。

33

2026.01.13

MySQL数据库报错常见问题及解决方法大全
MySQL数据库报错常见问题及解决方法大全

本专题整合了MySQL数据库报错常见问题及解决方法,阅读专题下面的文章了解更多详细内容。

18

2026.01.13

PHP 文件上传
PHP 文件上传

本专题整合了PHP实现文件上传相关教程,阅读专题下面的文章了解更多详细内容。

12

2026.01.13

PHP缓存策略教程大全
PHP缓存策略教程大全

本专题整合了PHP缓存相关教程,阅读专题下面的文章了解更多详细内容。

6

2026.01.13

jQuery 正则表达式相关教程
jQuery 正则表达式相关教程

本专题整合了jQuery正则表达式相关教程大全,阅读专题下面的文章了解更多详细内容。

3

2026.01.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Java 教程
Java 教程

共578课时 | 45.8万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号