std::atomic通过硬件支持实现共享变量的原子操作,避免竞态条件,并借助内存序控制线程间数据可见性与指令重排,常用memory_order_release与acquire保证同步,适用于简单变量的高效并发访问。

在C++多线程编程中,std::atomic 是实现线程安全操作的核心工具之一。它不仅能保证对共享变量的操作是原子的,还能控制内存访问顺序,确保数据在多个线程之间的可见性。理解其工作原理,有助于编写高效且正确的并发程序。
原子操作的基本保障
std::atomic 通过底层硬件支持来实现原子性。现代CPU提供了特定的原子指令(如x86的LOCK前缀指令、Compare-and-Swap等),编译器会将对 std::atomic 变量的操作编译为这些不可中断的机器指令。
例如,对一个 std::atomic
std::atomiccounter{0}; counter.fetch_add(1); // 原子加1
这个操作不会被其他线程打断,避免了竞态条件(race condition)。
立即学习“C++免费学习笔记(深入)”;
- 所有对 std::atomic 的读、写和修改操作都是原子的
- 无需使用互斥锁(mutex),减少开销
- 适用于整型、指针和部分自定义类型(需满足 trivially copyable 等条件)
内存可见性与内存序(Memory Order)
原子操作不仅解决原子性问题,还影响线程间的数据可见性。C++内存模型允许编译器和处理器对指令重排以优化性能,但这可能导致一个线程的修改无法及时被其他线程看到。
std::atomic 提供了六种内存序选项,用于控制操作的同步行为:
- memory_order_relaxed:仅保证原子性,不提供同步或顺序保证
- memory_order_acquire:用于读操作,确保后续读写不会被重排到该操作之前
- memory_order_release:用于写操作,确保之前的读写不会被重排到该操作之后
- memory_order_acq_rel:同时具备 acquire 和 release 语义
- memory_order_seq_cst:默认选项,提供最严格的顺序一致性,所有线程看到的操作顺序一致
例如,在生产者-消费者场景中:
std::atomicready{false}; int data = 0; // 生产者 data = 42; ready.store(true, std::memory_order_release); // 消费者 while (!ready.load(std::memory_order_acquire)) { // 等待 } // 此时能安全读取 data
这里使用 release 和 acquire 配对,确保 data = 42 不会被重排到 store 之后,从而保证消费者能看到正确的数据。
常见使用模式与注意事项
std::atomic 虽然强大,但使用时仍需注意以下几点:
- 不要假设所有类型都支持原子操作,非平凡类型可能不被支持
- 避免在原子变量上进行非原子操作,如先读再判断再写,应使用 compare_exchange_weak/strong
- 默认的 memory_order_seq_cst 安全但性能较低,高性能场景可考虑更宽松的内存序
- 原子操作不能替代锁处理复杂临界区,仅适合简单共享变量的同步
比如实现一个无锁计数器,可以直接使用 fetch_add;但如果涉及多个变量的复合逻辑,仍建议使用互斥量。
基本上就这些。掌握 std::atomic 的原子性和内存序机制,能帮助你在多线程环境中写出既高效又正确的代码。关键在于理解不同内存序的语义,并根据实际需求选择合适的模式。










