0

0

C++中的原子操作std::atomic有什么优势?(无锁并发与防止数据竞态)

穿越時空

穿越時空

发布时间:2026-01-10 09:35:16

|

438人浏览过

|

来源于php中文网

原创

std::atomic 保证单变量原子操作但不自动无锁,是否无锁取决于类型和平台,需用 is_lock_free() 检查;它不保证多变量一致性,不能替代 mutex 保护复杂状态,内存序必须显式指定。

c++中的原子操作std::atomic有什么优势?(无锁并发与防止数据竞态)

std::atomic 能避免数据竞态,但不等于自动无锁

直接说结论:std::atomic 的核心优势是提供**对单个变量的线程安全读写语义**,编译器和 CPU 会保证其操作不可分割(atomic),从而防止因指令重排或非原子读写导致的数据竞态。但它是否“无锁”,取决于具体类型和平台——std::atomic 在 x86-64 上通常编译为 lock xadd 等 CPU 原语,本质是硬件级锁;而 std::atomic<:shared_ptr>> 或大尺寸类型(如 std::atomic<:array>>)可能退化为内部互斥量实现,此时就是“有锁”的。

常见误判场景:

  • 以为所有 std::atomic 都是 lock-free:实际需检查 T::is_lock_free()std::atomic{}.is_lock_free()
  • ++fetch_add 安全更新计数器,却在多个 atomic 变量间做“逻辑组合”(如先读 A 再读 B),仍可能产生竞态——atomic 保单变量,不保多变量间的一致性

比 std::mutex 轻量,但适用场景很窄

std::atomic 的开销远低于 std::mutex:没有系统调用、无上下文切换、无阻塞等待。但它只解决“单个标量/POD 类型”的原子访问问题,无法替代锁来保护复杂状态或临界区。

典型适用场景包括:

立即学习C++免费学习笔记(深入)”;

  • 引用计数(std::shared_ptr 内部就重度依赖 std::atomic
  • 标志位开关(std::atomic 用于通知线程退出)
  • 计数器累加(std::atomic counter{0}; counter.fetch_add(1, std::memory_order_relaxed)
  • 无锁数据结构的基础构件(如 CAS 循环实现的 stack、queue)

一旦涉及多个字段协同更新(比如“余额减去金额,同时记录交易时间戳”),就必须用 std::mutex 或更高级同步原语,std::atomic 无法覆盖这类需求。

MetaVoice
MetaVoice

AI实时变声工具

下载

内存序(memory order)不是可选项,而是必选项

忽略内存序参数是 C++ 原子操作最隐蔽的坑。默认的 std::memory_order_seq_cst 最安全但性能最差;而 std::memory_order_relaxed 虽快,却允许编译器和 CPU 重排前后普通读写——这会导致看似正确的代码在多核上行为异常。

std::atomic ready{false};
int data = 0;

// 线程 A data = 42; // 普通写 ready.store(true, std::memory_order_relaxed); // 原子写,但无顺序约束

// 线程 B while (!ready.load(std::memory_order_relaxed)) { / 自旋 / } std::cout << data << "\n"; // 可能输出 0!因为 data=42 可能被重排到 ready.store 之后

正确做法是至少使用 std::memory_order_acquire / std::memory_order_release 配对:

// 线程 A
data = 42;
ready.store(true, std::memory_order_release); // 保证 data=42 不会重排到它之后

// 线程 B while (!ready.load(std::memory_order_acquire)) { / 自旋 / } std::cout << data << "\n"; // 此时 data=42 一定可见

std::atomic 的初始化和赋值容易踩坑

std::atomic 不支持拷贝构造和拷贝赋值,也不能用 = 直接赋值普通值(除非用 .store()隐式转换构造)。

  • 错误写法:std::atomic a = 5; —— 这是合法的,但只是调用隐式构造函数,不是赋值
  • 更危险的是:a = 5; 看似自然,实则调用 operator=(int),等价于 a.store(5),但容易让人误以为是普通赋值语义
  • 真正易错的是:未显式初始化的 std::atomic a; 是值不确定的(not zero-initialized),必须写成 std::atomic a{0};std::atomic a = ATOMIC_VAR_INIT(0);(C++17 起已弃用后者)
  • 聚合类型(如 struct)不能直接作为 std::atomic,除非 T 是 trivially copyable 且满足对齐要求;否则编译失败或运行时 fallback 到锁实现

真正难的从来不是“怎么写 atomic”,而是想清楚:这个变量是否真的只需原子读写?它的修改是否独立于其他状态?内存序是否匹配实际同步意图?这些问题没理清,std::atomic 只会让 bug 更难复现。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

533

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

51

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

193

2025.08.29

treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

533

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

17

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

12

2026.01.06

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

478

2023.08.10

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

25

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 6.4万人学习

C 教程
C 教程

共75课时 | 4万人学习

C++教程
C++教程

共115课时 | 11.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号