0

0

c++如何使用atomic原子操作_c++并发编程【进阶】

冰火之心

冰火之心

发布时间:2026-03-03 14:42:11

|

670人浏览过

|

来源于php中文网

原创

std::atomic 初始化必须显式,如 std::atomic counter{0}; 默认构造为未初始化状态;自定义类型需满足 is_trivially_copyable 等条件;load/store 的 memory_order 决定语义而非仅性能;fetch_add 等操作原子但不防溢出和 aba;atomic_ref 要求严格对齐与生命周期。

c++如何使用atomic原子操作_c++并发编程【进阶】

atomic 的初始化必须显式,不能依赖默认构造

很多初学者写 std::atomic<int> counter;</int> 后直接用 counter.load(),结果行为未定义——因为 std::atomic 的默认构造是“未初始化”状态,不保证内部值为 0。必须显式初始化:

  • std::atomic<int> counter{0};</int>(推荐,C++11 起支持)
  • std::atomic<int> counter = ATOMIC_VAR_INIT(0);</int>(C++17 已弃用,仅兼容旧代码)

对自定义类型(如 struct Point { int x,y; };),只有满足 is_trivially_copyable 且无虚函数/非静态成员引用时,才能用 std::atomic<point></point>;否则编译失败,得改用 std::atomic_ref 或锁。

load/store 的 memory_order 参数不是可选的“性能开关”,而是语义约束

counter.load() 看似简洁,实则等价于 counter.load(std::memory_order_seq_cst),这是最严格也最慢的顺序。但如果你只是读一个计数器用于日志打印,完全可以用 counter.load(std::memory_order_relaxed) 来避免不必要的内存栅栏。

  • std::memory_order_relaxed:只保证原子性,不约束前后指令重排 → 适合计数、标志位
  • std::memory_order_acquire:后续读写不能被重排到该 load 之前 → 常用于读取同步标志后访问共享数据
  • std::memory_order_release:前面读写不能被重排到该 store 之后 → 常用于写完数据再置位完成标志

混用 acquire/release 是成对出现的,单边乱设会导致数据竞争——比如生产者用 relaxed 写标志,消费者用 acquire 读,就无法建立 happens-before 关系。

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

Qwen
Qwen

阿里巴巴推出的一系列AI大语言模型和多模态模型

下载

fetch_add 等读-改-写操作天然线程安全,但要注意溢出和 ABA 问题

counter.fetch_add(1) 是原子的,返回旧值;counter++ 则不是(先 load 再 add 再 store,三步非原子)。但它不检查溢出:

  • std::atomic<uint32_t></uint32_t> 执行 fetch_add(1) 到 UINT32_MAX 后会回绕,不会抛异常
  • 若需饱和运算,得自己加锁或用 CAS 循环校验
  • ABA 问题在指针场景更突出:比如链表头指针被 A→B→A 修改两次,CAS 误判为未变;此时应改用 std::atomic<:shared_ptr>></:shared_ptr> 或带版本号的 tag-pointer 技巧

atomic_ref 是 C++20 引入的“零成本绑定”,但要求对象生命周期和对齐严格

当你有一块已存在的内存(比如全局数组、结构体字段),又不想把它改成 atomic<t></t> 类型时,std::atomic_ref 就派上用场了:

int arr[1024];
std::atomic_ref<int> ref{arr[5]}; // 绑定到 arr[5]
ref.store(42, std::memory_order_relaxed);

但必须满足两个硬性条件:

  • 被绑定对象的地址必须满足 alignof(T) 对齐(例如 int 通常需 4 字节对齐)
  • 对象生命周期必须长于 atomic_ref 实例——不能绑定栈变量然后返回 atomic_ref

不满足时行为未定义,且多数编译器不会报错,只在运行时崩溃或静默错误。

真正难的不是记住 memory_order 枚举值,而是判断哪段逻辑需要 acquire-release 配对、哪段可以放心用 relaxed;还有就是别把 atomic 当万能锁——它快,但表达能力有限,复杂同步还得靠 mutex 或 condition_variable。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

428

2025.06.09

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

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

201

2025.07.04

string转int
string转int

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

910

2023.08.02

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

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

598

2024.08.29

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

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

294

2025.08.29

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

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

210

2025.08.29

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

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

432

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

600

2023.08.10

Swift iOS架构设计与MVVM模式实战
Swift iOS架构设计与MVVM模式实战

本专题聚焦 Swift 在 iOS 应用架构设计中的实践,系统讲解 MVVM 模式的核心思想、数据绑定机制、模块拆分策略以及组件化开发方法。内容涵盖网络层封装、状态管理、依赖注入与性能优化技巧。通过完整项目案例,帮助开发者构建结构清晰、可维护性强的 iOS 应用架构体系。

0

2026.03.03

热门下载

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

精品课程

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

共94课时 | 10.5万人学习

C 教程
C 教程

共75课时 | 5.1万人学习

C++教程
C++教程

共115课时 | 20.2万人学习

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

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