0

0

c++如何使用std::atomic实现无锁编程 保证多线程数据安全【并发进阶】

冰火之心

冰火之心

发布时间:2026-01-14 08:41:02

|

681人浏览过

|

来源于php中文网

原创

std::atomic实现无锁编程需依赖硬件原子指令,适用于基础整型、指针及满足trivially copyable且lock-free的类型;须显式指定内存序,用compare_exchange_weak构建无锁结构,并规避ABA、内存泄漏等陷阱。

c++如何使用std::atomic实现无锁编程 保证多线程数据安全【并发进阶】

std::atomic 实现无锁编程,核心是避免互斥锁(如 std::mutex),靠硬件支持的原子操作保障读写安全。它不等于“完全不用同步”,而是把同步逻辑下沉到单个变量级别,由 CPU 指令(如 LOCK XCHGCMPXCHG)保证操作不可分割。

哪些类型适合用 std::atomic

基础整型(intlongbool)、指针,以及满足 trivially copyable 且大小适配原子指令的自定义类型(通常需 static_assert(std::atomic::is_always_lock_free) 验证)。浮点数虽有 std::atomic,但部分平台非 lock-free,性能可能打折扣。

  • 推荐优先使用 std::atomicstd::atomicstd::atomic
  • 避免对大结构体(如含 string 或 vector 的类)直接套 atomic —— 编译可能失败或退化为内部加锁
  • T.load()T.store(val) 替代直接读写,显式控制内存序

内存序(memory order)不是可选项,是关键设计点

默认 std::memory_order_seq_cst 最强,但开销最大;实际中常根据数据依赖关系降级以提升性能。例如:

  • 计数器累加:用 fetch_add(1, std::memory_order_relaxed) 足够 —— 只关心数值本身,不依赖其他变量
  • 生产者-消费者标志位:写端用 store(true, std::memory_order_release),读端用 load(std::memory_order_acquire),确保 release 前的写操作对 acquire 后的读可见
  • 自旋等待 flag 变成 true:用 while (!flag.load(std::memory_order_acquire)) { /* pause */ },配合 _mm_pause() 减少功耗

用 compare_exchange_weak 实现无锁/队列的核心技巧

这是无锁数据结构的基石。它做“比较并交换”:仅当当前值等于预期值时才更新,并返回是否成功。weak 版本允许虚假失败(spurious failure),需循环重试。

人民网AIGC-X
人民网AIGC-X

国内科研机构联合推出的AI生成内容检测工具

下载

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

  • 典型模式:do { expected = top.load(); new_node->next = expected; } while (!top.compare_exchange_weak(expected, new_node));
  • 注意:compare_exchange_weak 会修改 expected —— 失败时自动更新为当前真实值,下次循环直接重试
  • 避免在循环体内做复杂计算或分配内存,否则失败重试成本高
  • 若需强保证(如金融场景),可用 compare_exchange_strong,但多数情况 weak 更高效

常见陷阱与规避方式

无锁 ≠ 无脑快。错误假设易引发 ABA 问题、内存泄漏或死循环。

  • ABA 问题:指针被释放后又被新对象复用,导致 compare_exchange 误判成功。解法:用带版本号的 tag pointer(如 std::atomic 存高位版本+低位指针)
  • 忘记 delete 或智能指针管理:无锁结构中节点回收必须延迟(如 Hazard Pointer、RCU 或 epoch-based reclamation),不能立即 delete
  • atomic_flag 做自旋锁?可以,但它只支持 test-and-set,功能有限;更推荐直接用 std::atomic + exchange
  • 调试困难:原子操作无法打断点观察中间态,建议先用 std::mutex 验证逻辑,再逐步替换为 atomic

相关专题

更多
string转int
string转int

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

315

2023.08.02

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

84

2023.09.25

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

195

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

187

2025.07.04

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

195

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

187

2025.07.04

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是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

537

2024.08.29

php与html混编教程大全
php与html混编教程大全

本专题整合了php和html混编相关教程,阅读专题下面的文章了解更多详细内容。

3

2026.01.13

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
HTML5/CSS3/JavaScript/ES6入门课程
HTML5/CSS3/JavaScript/ES6入门课程

共102课时 | 6.7万人学习

前端基础到实战(HTML5+CSS3+ES6+NPM)
前端基础到实战(HTML5+CSS3+ES6+NPM)

共162课时 | 18.8万人学习

第二十二期_前端开发
第二十二期_前端开发

共119课时 | 12.4万人学习

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

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