原子内存顺序(memory_order)是C++多线程中控制原子操作重排序与可见性的机制,规定同步语义而非影响原子性;含relaxed、consume、acquire、release、acq_rel、seq_cst六种,需acquire-release配对建立synchronizes-with关系,核心是明确同步意图而非盲目选序。

原子内存顺序(memory_order)是 C++ 多线程中控制原子操作如何与非原子操作、其他原子操作之间进行重排序和可见性同步的机制。它不改变单个原子操作本身的原子性,而是规定该操作在内存模型中的“同步语义”——即其他线程何时能看到它的结果,以及编译器/CPU 能否将它与其他读写指令重排。
为什么需要 memory_order?
现代 CPU 和编译器会做大量优化:比如指令重排、缓存延迟写入、寄存器暂存等。这些对单线程无害,但在多线程下可能导致一个线程写入了变量,另一个线程却长期看不到更新(不是因为没执行,而是没同步到对方视角)。memory_order 就是用来显式告诉系统:“这里需要哪种程度的同步保障”。
六种 memory_order 及典型用途
memory_order_relaxed
只保证操作本身原子,不施加任何顺序约束。适合计数器、句柄生成等无需同步语义的场景。
例如:counter.fetch_add(1, std::memory_order_relaxed);
memory_order_consume(极少用,基本被弃用) 仅对依赖于该原子值的数据访问建立顺序(如指针解引用),实际中因难以正确使用且编译器支持弱,建议避免。
memory_order_acquire
用于读操作(如 load)。它保证:该读之后的所有读写操作,不会被重排到它前面;且能“看到”之前所有以 release 方式写入的值。
常见于锁的获取、标志位检查后读取关联数据。
memory_order_release
用于写操作(如 store)。它保证:该写之前的所有读写操作,不会被重排到它后面;其写入对配对的 acquire 读可见。
常见于锁的释放、设置完成标志前写入结果数据。
memory_order_acq_rel
用于读-修改-写操作(如 fetch_add, compare_exchange)。兼具 acquire 和 release 语义:操作前不重排,操作后不重排,且能同步对应 release/acquire 操作。
适合实现自旋锁、无锁栈的 push/pop。
memory_order_seq_cst(默认)
最强语义:所有线程看到的操作顺序一致,且每个原子操作都隐含 acquire + release + 全局顺序。性能开销最大,但最易理解、最安全。
除非有明确性能瓶颈并已分析清楚,否则优先用它。
acquire-release 配对才是关键
单独一个 acquire 或 release 没有意义,它们必须成对出现才能建立同步关系:
- 线程 A 用
store(x, memory_order_release)写入数据,再写flag.store(true, memory_order_release) - 线程 B 用
flag.load(memory_order_acquire)读到true,就能确保看到 A 写入的x的最新值 - 这就是“synchronizes-with”关系,是 C++ 内存模型中定义可见性的核心机制
容易踩的坑
把 relaxed 用在本该同步的地方(比如状态标志),会导致未定义行为或偶发 bug,极难复现。
混用不同 order 时没形成 acquire-release 链,同步就断了。
误以为 seq_cst 能防止所有重排(它不能禁止非原子操作之间的重排,也不能替代互斥锁保护临界区)。
基本上就这些。理解 memory_order 的关键是:它不是“让操作变快/变慢”,而是“划定哪些操作必须按什么顺序对其他线程可见”。写多线程代码时,先想清楚同步意图,再选合适的 order,比死记规则更重要。
立即学习“C++免费学习笔记(深入)”;










