C++(26): 原子操作(std::atomic)

目录

1. 简述

2. 什么是原子操作

3. C++原子操作

4. std::atomic_flag

5. std::atomic

(1)操作

(2)赋值(store)、读取(load)与交换(exchange)

(3)算术运算

(4)CAS


1. 简述

        开发过多线程、并发编程的小伙伴一定接触过mutex,通过对资源进行加锁和解锁,实现对方问和修改的互斥操作。mutex使用起来很方便,很强大,但也有局限性。频繁地加锁和解锁会造成较大的资源消耗,影像系统的性能。

        与mutex相比,原子(atomic)操作相对灵活和简单。

        注意,这种灵活性在一定程度上是做了某些妥协的。

2. 什么是原子操作

        原子指的是一系列不可被CPU上下文交换的机器指令,这些指令组合在一起就形成了原子操作。

        我们在日常使用的CPU或SOC基本都是多核的情况,当其中某个CPU核心开始运行原子操作时,会先暂停其它CPU内核对内存的操作,保证对资源的独占性,进而保证资源不会被其它CPU内核所干扰,这就是原子操作的通俗解释。

3. C++原子操作

        C++提供了一个模板类型std::atomic<T>来助力实现原子操作,还提供了一些特化的原子类型,例如std::atomic_int、std::atomic_long等。

        此外还提供了std::atomic_flag这一超简单的原子类型,简单到只有设置(set)和清除(clear)两种状态。

4. std::atomic_flag

        std::atomic_flag可以说得上是最简单的原子类型了,他只有设置(set)和清除(clear)两种状态。std::atomic_flag不可拷贝和赋值,且必须使用ATOMIC_FLAG_INIT宏初始化。

#include <atomic>std::atomic_flag flag = ATOMIC_FLAG_INIT;

        关于atomic_flag我们只需要掌握两个成员的使用就可以了,他们分别是test_and_set和clear。

        test_and_set用于判断当前变量是否被设置过,如果没有被设置过,则进行设置,并返回false,反之则直接返回true。

        clear用于清除设置的状态。

        如下所示的例程演示了排他性的访问某些资源,创建10个线程,分别访问同一段资源。当test_and_set返回true时,说明有其他某个线程正在访问,因此等待,知道test_and_set返回false,进行访问,之后清除。

#include <iostream>#include <atomic>#include <thread>#include <vector>#include <sstream>std::atomic_flag atomic_state = ATOMIC_FLAG_INIT;std::stringstream stream_info;void access_stream(int x){while (atomic_state.test_and_set()); ///< 等待状态被清除stream_info << "thread" << x << "access stream" << '\n';atomic_state.clear(); ///< 清楚状态}int main(){std::vector < std::thread > threads;for(int i = 1; i <= 10; ++i){threads.push_back(std::thread(access_stream, i));}for(auto & th:threads){th.join();}std::cout << stream_info.str() << std::endl;;return 0;}

5. std::atomic<T>

        std::atomic<T>作为一个模板,提供了通用的原子类型,也提供了比std::atomic_flag更为灵活和复杂的应用功能。

        std::atomic_int等作为特化的原子类型,是特殊的std::atomic<T>,一般来讲std::atomic_int等价于std::atomic<int>,其他特化类型类似。

(1)操作

        std::atomic提供了赋值、算术运算和比较交换等操作。

(2)赋值(store)、读取(load)与交换(exchange)

        std::tomic提供了store和load接口,分别用来赋值和读取,也提供了exchange用来交换新值,返回旧值。

#include <iostream>#include <atomic>int main(int argc, char* argv[]){std::atomic<int> atomic_int(0);atomic_int.store(10); ///< 设置原子变量的值std::cout << "value: " << atomic_int.load() << std::endl;int value = atomic_int.load(); ///< 读取原子变量的值std::cout << "value: " << value << std::endl;int old_value = atomic_int.exchange(20); // 交换原子变量的值std::cout << "old_value: " << old_value << ", new_value: " << atomic_int.load() << std::endl;return 0;}

(3)算术运算

        std::atomic提供了原子加,原子减等接口,具体包含fetch_add、fetch_sub、fetch_and、fetch_or和fetch_xor等。

        原子算术运算后,都会返回原值。

#include <iostream>#include <atomic>int main(int argc, char* argv[]){std::atomic<int>    atomic_int(0);int last_value;last_value = atomic_int.fetch_add(10); ///< 原子加操作 std::cout << "last_value: " << last_value << ", new_value: " << atomic_int.load() << std::endl;last_value = atomic_int.fetch_sub(5); ///< 原子减操作std::cout << "last_value: " << last_value << ", new_value: " << atomic_int.load() << std::endl;last_value = atomic_int.fetch_and(0b1100); ///< 原子与操作std::cout << "last_value: " << last_value << ", new_value: " << atomic_int.load() << std::endl;last_value = atomic_int.fetch_or(0b1010); ///< 原子或操作std::cout << "last_value: " << last_value << ", new_value: " << atomic_int.load() << std::endl;last_value = atomic_int.fetch_xor(0b1111); ///< 原子异或操作std::cout << "last_value: " << last_value << ", new_value: " << atomic_int.load() << std::endl;return 0;}

(4)CAS

        CAS,Compare and Swap,比较并交换。

        std::atomic提供了compare_exchange_weak和compare_exchange_strong实现比较及交换功能,二者的功能是一样的,但是前者性能更好一些,常在高速循环中使用。

        参数传入期待值与新值,通过比较当前值与期待值的情况进行区别改变。

a.compare_exchange_weak(b, c)其中a是当前值,b期望值,c新值

a==b时:函数返回真,并把c赋值给a

a!=b时:函数返回假,并把a复制给b

#include <iostream>#include <atomic>int main(int argc, char* argv[]){std::atomic<int> a;a.store(10);int b=10;int c=20;std::cout<<"a:"<<a<<std::endl;if(a.compare_exchange_weak(b, c)){ ///< a和b值相同,把c的值赋给astd::cout<<"a true:"<<a.load()<<std::endl;}std::cout<<"a:"<<a<<" b:"<<b<<" c:"<<c<<std::endl;return 0;}>> 运行结果a: 20 b:10 c:20

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/31409.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java学习笔记(一)Java内容介绍、程序举例、DOS命令、Java跨平台特性的本质

Hi i,m JinXiang ⭐ 前言 ⭐ 本篇文章主要介绍Java内容介绍、程序举例、DOS命令、Java跨平台特性的本质详细介绍以及部分理论知识 🍉欢迎点赞 👍 收藏 ⭐留言评论 📝私信必回哟😁 🍉博主收将持续更新学习记录获,友友们有任何问题可以在评论区留言 目录 1、内容介绍…

STM32学习和实践笔记(37):DMA实验

1.DMA简介 DMA&#xff0c;全称是Direct Memory Access&#xff0c;中文意思为直接存储器访问。DMA可用于实现外设与存储器之间或者存储器与存储器之间数据传输的高效性。 之所以高效&#xff0c;是因为DMA传输数据移动过程无需CPU直接操作&#xff0c;这样节省的 CPU 资源就可…

小白教程:使用IntelliJ IDEA的HTTP Client进行接口验证

问题背景 这段时间使用开发一些Rest API相关的功能&#xff0c;准备做一些接口的简单测试&#xff0c;快速的验证一下API功能是否正常&#xff0c;正好觉得IntelliJ IDEA中的HTTP Client功能非常方便&#xff0c;它允许我们直接在编辑器中操作&#xff0c;正好记录一下。 解决…

小程序使用接口wx.getLocation配置

开通时需详细描述业务&#xff0c;否则可能审核不通过 可能需要绑定腾讯位置服务&#xff0c;新建应该&#xff0c;绑定到小程序 配置 权限声明&#xff1a;在使用wx.getLocation前&#xff0c;需要在app.json的permission字段中声明对用户位置信息的使用权限&#xff0c;并提…

掘金淘宝API:揭秘店铺商品详情的智能获取秘籍

引言 在浩瀚的电商世界里&#xff0c;淘宝作为中国的电商巨头&#xff0c;不仅为买家提供了琳琅满目的商品选择&#xff0c;更为开发者开启了数据宝藏的大门。通过淘宝API&#xff0c;你能够轻松获取店铺所有商品的详尽信息&#xff0c;无论是为了市场分析、价格监控&#xff…

大数据学习-大数据介绍

意义 从海量的数据中分析出海量数据背后的价值 需要分析海量的数据&#xff0c;就需要存储、计算和分析 那就需要分布式多台计算机合适的工具来处理数据 工具 特点 大数据的核心工作&#xff1a;从海量的、高增长的、多类别的、信息密度低的数据中挖掘出高质量的结果 数据存储…

【CS.AI】决策树算法介绍: 原理与案例实现

文章目录 1. 简介1.1 决策树的基本原理核心概念: 2. 决策树的应用案例2.1 金融行业中的信用评分2.2 医疗诊断中的应用2.3 零售行业中的市场营销 3. 决策树的优缺点优点:缺点: 4. Python代码示例5 结论 [toc] ![在这里插入图片描述 1. 简介 决策树是一种常见且强大的机器学习算…

!=和!==的区别

在JavaScript中&#xff0c;! 和 ! 是两种不同类型的比较运算符&#xff0c;它们之间的主要区别在于它们如何处理比较中的类型转换&#xff08;也称为“强制类型转换”或“宽松比较”与“严格比较”&#xff09;。 !&#xff08;宽松比较&#xff09;: 如果两个操作数的类型不同…

本地部署 ChatTTS

本地部署 ChatTTS 0. ChatTTS 简介1. ChatTTS 亮点2. 创建虚拟环境3. 克隆代码4. 安装依赖5. 快速开始6. 访问 0. ChatTTS 简介 ChatTTS 是一款专门为对话场景&#xff08;例如 LLM 助手&#xff09;设计的文本转语音模型。 1. ChatTTS 亮点 对话式 TTS: ChatTTS 针对对话式…

基坑监测的内容及其重要性概述

随着城市建设的不断深入&#xff0c;基坑工程作为基础建设的重要组成部分&#xff0c;其安全性和稳定性成为了关注的重点。为了确保基坑施工过程中的安全&#xff0c;基坑监测显得尤为重要。本文将围绕基坑监测的内容展开&#xff0c;旨在帮助读者更好地理解其重要性及实施方法…

卫星导航与gazebo仿真

全球卫星导航系统(Global Navigation Satelite System,GNSS)&#xff0c;简称卫星导航&#xff0c;是室外机器人定位的一个主要信息来源。 卫星导航能给机器人提供什么信息&#xff1f; 正常工作时&#xff0c;实际上可以提供机器人所需的所有定位信息&#xff0c;包括&#x…

用了这么久的群晖NAS,它到底能干些什么?

从21年开始玩群晖也有几年了&#xff0c;除非面临断电或升级&#xff0c;这个小伙伴都任劳任怨的工作着 现在NAS也广泛应用于家庭和企业环境中了&#xff0c;今天盘点一下我用群晖NAS都干了些什么~ 1.文件存储与共享&#xff1a; 群晖NAS可以作为文件服务器&#xff0c;提供…

Windows——报错解决:Linux服务器下载的文件夹打不开

问题描述&#xff1a; 显示已经占用了内存&#xff0c;但是点进文件夹报错。 解决办法&#xff1a; Linux服务器上使用zip压缩后&#xff0c;然后下载到windows电脑&#xff0c;然后解压。

【Autoware】Autoware.universe安装过程与问题记录

&#x1f60f;★,:.☆(&#xffe3;▽&#xffe3;)/$:.★ &#x1f60f; 这篇文章主要介绍Autoware.universe安装过程与问题记录。 无专精则不能成&#xff0c;无涉猎则不能通。——梁启超 欢迎来到我的博客&#xff0c;一起学习&#xff0c;共同进步。 喜欢的朋友可以关注一下…

七人拼团:互助共赢,电商新动力

在当前繁荣的电商领域中&#xff0c;七人互助拼团模式以其别具一格的激励机制和互助合作理念&#xff0c;成为了消费者和商家共同瞩目的焦点。接下来&#xff0c;我们将详细解读这一模式中的直推激励、滑落补偿以及团队成就奖&#xff0c;并探讨其如何体现互助合作的精神。 一、…

190.回溯算法:组合(力扣)

代码随想录 (programmercarl.com) 一、什么是回溯算法 回溯算法是一种通用的算法设计技巧&#xff0c;特别适用于解决组合、排列、子集等问题。它通过逐步构建解决方案&#xff0c;并在发现部分解决方案无效时撤销&#xff08;回溯&#xff09;部分计算&#xff0c;从而寻找所…

灵活的招聘管理系统有五种方法帮助成功招聘

还记得以前的时代吗&#xff1f;这取决于你的年龄&#xff0c;直到智能手机、流媒体电视和电子邮件出现。今天&#xff0c;任何活着的成年人都经历了技术上的巨大变化&#xff0c;这创造了一种新的行为方式。人才获取也是如此。 一个值得推荐的招聘管理系统 招聘团队被困在满足…

zlib库的交叉编译记录

zlib库的交叉编译记录 嵌入式项目中要用到zlib库&#xff0c;今天下载交叉编译了一遍&#xff0c;发现和其它库有点区别&#xff0c;这里记录一下。 1.首先clone到本地 git clone https://github.com/madler/zlib.git2.建立一个安装目录 mkdir ~/zlib-arm-install3.声明一个…

C# —— 构造函数

什么是构造函数 构造函数: 一般在函数为类的属性初始值的作用&#xff0c;构造函数的名称类名 在类里面定义构造函数 方法名和类名同名 不能带返回值类型 void/非void 不能有 // 创建一个构造函数 class People {public string Name { get; set; }public int Age { get; set;…

用于制作耳机壳的UV树脂耳机壳UV胶价格高不高?

用于制作耳机壳的UV树脂耳机壳UV胶价格高不高&#xff1f; 制作耳机壳的UV树脂价格相对于一些其他材料可能会略高&#xff0c;但具体的价格取决于多个因素&#xff0c;如品牌、型号、质量等。一些高端的UV树脂品牌和型号可能会价格较高&#xff0c;但它们也通常具有更好的性能…