memory_order 是 C++ 原子操作中显式指定内存访问顺序约束的关键参数,控制周边非原子内存访问的重排边界并影响多线程间修改的可见性时机,本质是告知编译器和 CPU 哪些指令重排被允许或禁止。

memory_order 是 C++ 原子操作中用于**显式指定内存访问顺序约束**的关键参数,它不改变原子变量本身的读写行为,而是控制该原子操作**周边普通内存访问(非原子)的重排边界**,并影响多线程间对内存修改的**可见性时机**。核心在于:它不是“让操作变快/慢”,而是“告诉编译器和 CPU 哪些重排可以做、哪些必须禁止”。
为什么需要 memory_order?
现代 CPU 和编译器为优化性能,会重排指令(如把后面读内存提前、把前面写延后)。单线程下这没问题,但多线程共享内存时,重排可能导致其他线程看到“不合逻辑”的中间状态。比如:
- 线程 A 写数据 data = 42,再写标志 ready = true
- 线程 B 看到 ready == true,却读到 data == 0(因为写 data 被重排到了 ready 后面)
memory_order 就是用来插入“内存屏障(fence)”,阻止这类有害重排,建立线程间的同步关系。
六种 memory_order 及典型用途
memory_order_relaxed:无同步、无顺序约束。仅保证原子性,不阻止任何重排。适合计数器、引用计数等无需同步的场景。
立即学习“C++免费学习笔记(深入)”;
memory_order_acquire:用于读操作(load)。保证该读之后的所有内存访问(读/写)不能被重排到该读之前。常用于“获取锁”或“读就绪标志”。
memory_order_release:用于写操作(store)。保证该写之前的所有内存访问不能被重排到该写之后。常用于“释放锁”或“写就绪标志”。
memory_order_acq_rel:用于读-修改-写操作(如 fetch_add、compare_exchange)。兼具 acquire 和 release 语义,前后都设屏障。
memory_order_seq_cst(默认):最严格。所有带此序的原子操作构成一个全局单一执行顺序,且每条都具备 acquire + release 语义。简单安全,但可能有性能开销。
memory_order_consume:已基本弃用。理论上只约束依赖链上的重排(如指针解引用),实际难以正确使用,主流编译器按 acquire 处理。
acquire-release 配对是核心同步模式
这是最常用、高效且易理解的同步方式:
- 线程 A 执行 flag.store(true, memory_order_release) → 把之前所有写(如 data = 42)“发布”出去
- 线程 B 执行 if (flag.load(memory_order_acquire)) { use(data); } → 一旦看到 flag,就能确保看到 A “发布”的全部内容
- 二者不需在同一个变量上配对;不同原子变量也可形成同步关系(只要存在 happens-before)
使用建议与注意事项
- 初学者优先用 memory_order_seq_cst,逻辑清晰、不易出错
- 性能敏感路径再考虑 acquire/release,但务必成对出现、语义匹配
- 避免混用不同序在同一变量上(如对同一 atomic
有时用 relaxed,有时用 seq_cst),易破坏预期顺序 - memory_order 不解决数据竞争本身——变量仍需是 atomic 类型;它解决的是“看到什么”和“何时看到”










