0

0

如何在C++中利用原子操作(Atomic)避免锁竞争?(无锁编程)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-03-12 20:23:35

|

252人浏览过

|

来源于php中文网

原创

应优先选用标准原子整型别名(如std::atomic_int),避免对自定义结构体直接套用std::atomic;memory_order按需选relaxed/release/acquire而非全用seq_cst;循环CAS必须用compare_exchange_weak;ABA问题需用tagged pointer等方案规避。

如何在c++中利用原子操作(atomic)避免锁竞争?(无锁编程)

std::atomic 该选什么类型?别直接套用 int

原子变量不是万能胶水,std::atomic<int></int> 能用,但多数真实场景下它掩盖了对齐和大小的隐患。比如在结构体里嵌套 std::atomic<bool></bool>,某些平台可能因内存布局导致非原子读写;又比如用 std::atomic<long></long> 在 32 位 Windows 上,实际是锁模拟(InterlockedExchange64),性能反不如手写自旋。

实操建议:

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

  • 优先用标准原子整型别名:std::atomic_intstd::atomic_uintptr_t,它们保证底层操作是 lock-free 的(可通过 .is_lock_free() 检查)
  • 避免对自定义结构体直接套 std::atomic<mystruct></mystruct> —— 即使编译通过,大概率 fallback 到内部互斥锁,失去无锁意义
  • 若需原子更新多个字段(如 x/y 坐标),不要拼 std::atomic<uint64_t></uint64_t> 手动位拆分,改用 std::atomic<:pair int>></:pair> 并确认 is_lock_free() 返回 true

memory_order 怎么选?别全用 memory_order_seq_cst

memory_order_seq_cst 是默认且最安全的顺序,但它强制全局顺序,在多核 CPU 上会插入大量内存屏障(如 mfence),吞吐量可能比 relaxed 模式低 3–5 倍。而多数计数器、状态标志根本不需要这么强的约束。

实操建议:

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

  • 计数器累加(如引用计数):用 memory_order_relaxed,只要不依赖其他内存访问顺序
  • 生产者-消费者信号(如 flag 变成 true 表示数据就绪):写端用 memory_order_release,读端用 memory_order_acquire,确保 flag 之前的写操作对读端可见
  • 绝对避免在循环中反复调用 load(memory_order_seq_cst) 等待条件 —— 它会阻止编译器优化,且每次都刷全核缓存一致性协议

compare_exchange_weak 和 strong 什么时候换?

compare_exchange_weak 可能在无竞争时也失败(spurious failure),尤其在 ARM 或旧版 x86 上;compare_exchange_strong 保证只在值不匹配时失败。但 weak 版本生成的汇编更轻(比如 ARM 的 ldxr/stxr 对),循环重试开销更低。

绘蛙
绘蛙

电商场景的AI创作平台,无需高薪聘请商拍和文案团队,使用绘蛙即可低成本、批量创作优质的商拍图、种草文案

下载

实操建议:

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

  • 所有循环 CAS 场景(如无锁栈 push/pop)必须用 compare_exchange_weak,配合 do-while,这是标准写法
  • 单次尝试、不能重试的逻辑(如初始化全局原子指针)才考虑 compare_exchange_strong
  • 别在 weak 失败后直接 return 或 log —— 它失败不代表冲突,大概率是硬件抖动,应继续循环

无锁 ≠ 无风险:ABA 问题怎么绕过去?

典型现象:线程 A 读到指针 p = 0x1000,被抢占;线程 B 把 p 指向的对象释放,又新分配到同一地址 0x1000 并修改内容;A 恢复后执行 compare_exchange 成功,却误以为对象没变 —— 数据被静默覆盖。

实操建议:

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

  • 简单计数器场景(如 tag + ptr 组合):用 std::atomic<:shared_ptr>></:shared_ptr>,其内部已做 tag 处理(部分实现)
  • 手动解决:把指针和版本号打包进 uint64_t,高位存版本(每次释放+1),低位存指针,用 compare_exchange_weak 原子更新整个 64 位值
  • 别指望 std::atomic_flag 能躲开 ABA —— 它连 load 都不支持,纯用于自旋锁基元,和业务逻辑无关

真正难的从来不是写对一个 fetch_add,而是厘清哪些变量之间存在隐含依赖、哪些顺序约束被你无意中删掉了。稍不注意,lock-free 就退化成 lock-heavy。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
while的用法
while的用法

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

107

2023.09.25

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

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

490

2025.06.09

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

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

202

2025.07.04

string转int
string转int

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

1030

2023.08.02

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

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

612

2024.08.29

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

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

334

2025.08.29

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

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

235

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

443

2023.07.18

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

76

2026.03.11

热门下载

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

精品课程

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

共94课时 | 11.2万人学习

C 教程
C 教程

共75课时 | 5.4万人学习

C++教程
C++教程

共115课时 | 21.6万人学习

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

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