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<int></int> 在 x86-64 上通常编译为 lock xadd 等 CPU 原语,本质是硬件级锁;而 std::atomic<:shared_ptr>></:shared_ptr> 或大尺寸类型(如 std::atomic<:array>></:array>)可能退化为内部互斥量实现,此时就是“有锁”的。

常见误判场景:

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

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

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

典型适用场景包括:

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

Veed AI Voice Generator
Veed AI Voice Generator

Veed推出的AI语音生成器

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

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

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

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

std::atomic<bool> ready{false};
int data = 0;
<p>// 线程 A
data = 42;                              // 普通写
ready.store(true, std::memory_order_relaxed); // 原子写,但无顺序约束</p><p>// 线程 B
while (!ready.load(std::memory_order_relaxed)) { /<em> 自旋 </em>/ }
std::cout << data << "\n"; // 可能输出 0!因为 data=42 可能被重排到 ready.store 之后</p>

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

// 线程 A
data = 42;
ready.store(true, std::memory_order_release); // 保证 data=42 不会重排到它之后
<p>// 线程 B
while (!ready.load(std::memory_order_acquire)) { /<em> 自旋 </em>/ }
std::cout << data << "\n"; // 此时 data=42 一定可见</p>

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

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

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

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

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

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

970

2023.08.02

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

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

605

2024.08.29

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

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

294

2025.08.29

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

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

212

2025.08.29

treenode的用法
treenode的用法

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

548

2023.12.01

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

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

27

2025.12.22

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

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

44

2026.01.06

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

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

763

2023.08.10

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

23

2026.03.06

热门下载

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

精品课程

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

共94课时 | 10.9万人学习

C 教程
C 教程

共75课时 | 5.3万人学习

C++教程
C++教程

共115课时 | 21万人学习

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

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