std::atomic是C++11提供的模板类,用于封装变量并保证其操作的原子性,如int、bool、指针等类型;通过load、store、fetch_add等操作实现线程安全的共享变量访问,避免数据竞争和锁带来的性能开销;常用于无锁编程场景,如计数器累加,提升并发效率。

在C++11中,std::atomic 是实现无锁编程的核心工具之一。它提供了一种线程安全的方式来操作共享变量,而无需使用互斥锁(mutex)。这不仅提升了性能,还避免了死锁、锁竞争等问题。
什么是 std::atomic?
std::atomic 是一个模板类,用于封装某种类型的变量(如 int、bool、指针等),保证对该变量的读取和写入操作是原子的——即不会被其他线程打断。
常见的用法包括:
std::atomiccounter{0}; std::atomicready{false}; std::atomicptr;
一旦声明为 atomic 类型,对它的 ++、--、load()、store()、fetch_add() 等操作都具备原子性。
立即学习“C++免费学习笔记(深入)”;
为什么需要 atomic 实现无锁编程?
在多线程环境中,多个线程同时修改同一个变量可能导致数据竞争(data race),从而引发未定义行为。传统做法是用 mutex 加锁保护共享资源,但加锁会带来开销。
使用 std::atomic 可以避免锁的使用,在一些简单场景下实现高效、安全的并发访问。
例如,计数器累加可以用 atomic 替代锁:
std::atomiccount{0}; void increment() { for (int i = 0; i < 1000; ++i) { count.fetch_add(1, std::memory_order_relaxed); } }
多个线程调用 increment() 不会出现竞争,且性能远高于加锁版本。
内存顺序:控制原子操作的可见性与顺序
atomic 操作支持不同的内存顺序(memory order),用来平衡性能与同步需求。常用的有:
- std::memory_order_relaxed:只保证原子性,不保证顺序(适合计数器)
- std::memory_order_acquire:用于 load,确保后续读写不会被重排到该操作之前
- std::memory_order_release:用于 store,确保前面的读写不会被重排到该操作之后
- std::memory_order_acq_rel:acquire + release,用于读-改-写操作
- std::memory_order_seq_cst:最严格的顺序一致性,默认选项
合理选择 memory order 能提升性能,但过度放松可能导致逻辑错误,需谨慎使用。
atomic 的典型应用场景
std::atomic 特别适用于以下情况:
-
标志位控制:如用 atomic
控制线程是否退出 - 高性能计数器:日志统计、请求计数等高频更新场景
- 无锁队列/栈的基础构件:配合 CAS(compare_exchange_weak/strong)实现复杂结构
- 单例模式中的双重检查锁定:确保初始化只执行一次
比如用 compare_exchange_strong 实现原子更新:
std::atomicvalue{0}; int expected = value.load(); while (!value.compare_exchange_strong(expected, 42)) { // 如果当前值不等于 expected,则 compare_exchange 更新 expected 为当前值 // 并返回 false,循环继续 }
基本上就这些。std::atomic 是 C++11 提供的轻量级同步机制,掌握它有助于写出高效、安全的并发代码。关键是理解原子操作的语义和内存模型,避免误用导致隐蔽 bug。











