std::atomic是c++11引入的模板类,用于实现共享数据的原子操作,确保多线程环境下对变量的访问不会引发数据竞争。它支持int、bool、指针等可平凡复制类型,提供load、store、fetch_add、compare_exchange_weak等原子操作,具有不可分割性、内存顺序可控、无数据竞争等特点,常用于线程安全计数器、无锁编程等场景,通过合理使用可提升并发性能。

在C++多线程编程中,std::atomic 是保证共享数据原子访问的核心工具。它用于封装一个类型的数据,使得对该数据的读取和写入操作不会被多个线程同时修改导致数据竞争,从而实现线程安全。
什么是 std::atomic?
std::atomic 是C++11引入的一个模板类,定义在 <atomic> 头文件中。它可以包装诸如 int、bool、指针等基本类型,确保对这些变量的操作是“原子的”——即不会被中断,也不会出现中间状态。
例如:
std::atomic<int> counter{0};
立即学习“C++免费学习笔记(深入)”;
多个线程同时执行 counter++ 时,不会发生竞态条件,结果始终正确。
原子操作的基本特性
原子操作的关键在于“不可分割性”。以下是其核心特点:
- 无数据竞争:只要所有线程都通过 atomic 操作访问变量,就不会出现未定义行为。
- 内存顺序可控:可通过指定内存序(如 memory_order_relaxed、memory_order_seq_cst)控制操作的同步方式,平衡性能与一致性。
- 支持常见操作:包括 load、store、fetch_add、exchange、compare_exchange_weak 等。
示例:使用 fetch_add 实现线程安全计数器
std::atomic<int> value{0};
// 在某个线程中:
value.fetch_add(1); // 原子地加1
compare-and-swap 与无锁编程
compare_exchange_weak 和 compare_exchange_strong 是实现无锁数据结构的关键。
std::atomic<int> data;
int expected = data.load();
do {
if (data.compare_exchange_weak(expected, new_value))
break;
} while (true);
这种模式常用于实现线程安全的单例、无锁队列等高级结构。
哪些类型可以用于 std::atomic?
std::atomic 可用于所有可平凡复制(trivially copyable)的类型。标准库为部分基本类型提供了特化版本:
注意:不是所有类型都支持完整的原子操作。复杂类型可能只支持 load/store,而不支持算术操作。
基本上就这些。std::atomic 提供了高效、低开销的线程同步手段,尤其适用于计数器、状态标志、引用计数等场景。合理使用原子操作,能避免互斥锁的开销,提升并发性能,但也要注意内存序的选择,避免因过度放松顺序导致逻辑错误。


