memory_order 是 C++ 原子操作的内存序约束,不改变读写语义,仅限制普通内存访问重排及跨线程可见性;选错会导致隐式数据竞争。五种序按“重排+可见性”强度递增:relaxed(无约束)、acquire(后序不可前移)、release(前序不可后移)、acq_rel(二者兼具)、seq_cst(全局顺序一致,默认且最安全)。

memory_order 是 C++ 原子操作的内存序约束标记
它不改变原子操作本身的读/写语义,而是告诉编译器和 CPU:这个原子操作周围,普通内存访问(非原子)可以怎么重排、其他线程看到该操作结果的时机如何保证。选错 memory_order 会导致数据竞争或逻辑错误,但不会直接报错——这是最危险的地方。
五种 memory_order 的实际行为差异
重点不是背名字,而是看它们对「重排」和「可见性」的限制强度:
-
memory_order_relaxed:只保证原子性,不约束重排,也不建立同步关系。适合计数器、引用计数等纯数值场景 -
memory_order_acquire:当前操作之后的所有读写(包括普通变量)不能被重排到它前面;用于“读取一个 flag 后安全读取其保护的数据” -
memory_order_release:当前操作之前的所有读写(包括普通变量)不能被重排到它后面;用于“写完数据后设置 flag” -
memory_order_acq_rel:同时具备 acquire 和 release 行为,仅用于 read-modify-write 操作(如fetch_add) -
memory_order_seq_cst:最强,默认行为。所有线程看到的原子操作顺序一致,且隐含全局顺序;性能开销最大,但最不容易出错
典型错误:用 relaxed 读 flag 却依赖后续数据
常见于“双检锁”或“懒初始化”场景。比如:
std::atomicready{false}; int data = 0; // 线程 A data = 42; ready.store(true, std::memory_order_relaxed); // ❌ 错误:data 可能还没写入就设了 ready // 线程 B if (ready.load(std::memory_order_relaxed)) { // ❌ 错误:即使看到 true,data 仍可能为 0 std::cout << data << "\n"; // 可能输出 0 }
正确做法是:
立即学习“C++免费学习笔记(深入)”;
- A 中用
ready.store(true, std::memory_order_release) - B 中用
ready.load(std::memory_order_acquire)
这样就能保证 A 写 data 的动作对 B 可见。
性能与可读性的实际权衡点
在 x86 上,acquire/release 几乎不产生额外指令(靠 lfence/sfence 的缺失实现),而 seq_cst 需要 mfence 或带锁指令;ARM/AArch64 则差异明显。但更关键的是:除非你明确知道 relaxed 不会破坏逻辑,否则优先用 memory_order_acquire/memory_order_release 配对,比盲目上 seq_cst 更合理——后者掩盖问题,前者暴露设计缺陷。
真正容易被忽略的是:哪怕只改一个 load 的 memory_order,也可能让整个同步链失效;它从来不是孤立参数,而是成对出现的契约。










