C++ 并发专题 - 原子等待(atomic wait)

一:概述

        原子等待是 C++20 引入的一个特性,主要用于多线程编程中。它允许线程高效地等待某个原子变量的值变化,而不是使用忙等待(busy waiting),这样可以减少 CPU 资源的浪费,提高程序的性能。

        原子等待的引入主要是为了提高性能和资源利用率,尤其是在以下场景中:

  1. 减少上下文切换:传统的等待机制(如互斥锁、条件变量)常常涉及线程的阻塞与唤醒,这会导致上下文切换的开销。原子等待则允许线程在等待时进入休眠状态,避免了这些开销。

  2. 提高效率:在多线程环境中,频繁的状态变化可能会导致大量的线程等待和唤醒,使用原子等待可以更高效地管理这些状态变化,降低 CPU 的负担。

  3. 简化实现:原子等待提供了一种简单的机制,使得开发者可以更方便地实现高效的无锁数据结构和算法。

  4. 适用于高竞争场景:在高竞争的场景中,传统的锁机制可能会导致严重的性能下降,而原子等待能够减少锁竞争带来的问题。

二:多线程编程中还有哪些等待机制:

        在多线程编程中,除了原子等待之外,还有几种常见的等待机制,每种机制都有其特定的使用场景和优缺点:

1. 忙等待(Busy Waiting)

  • 描述:线程循环检查某个条件是否满足,一直占用 CPU。

  • 优点:实现简单,延迟较低。

  • 缺点:浪费 CPU 资源,导致性能下降,尤其在高负载时。

2. 互斥锁(Mutex)

  • 描述:使用互斥锁控制对共享资源的访问,线程在获取不到锁时会阻塞。

  • 优点:适用于复杂的临界区,能有效防止数据竞争。

  • 缺点:锁的开销较大,可能导致死锁、优先级反转等问题。

3. 条件变量(Condition Variable)

  • 描述:与互斥锁配合使用,线程可以在条件变量上等待,当条件满足时被通知。

  • 优点:避免了忙等待,可以在条件满足时高效唤醒线程。

  • 缺点:相对复杂,需要正确管理互斥锁和条件变量。

4. 信号量(Semaphore)

  • 描述:允许多个线程访问共享资源的计数器。线程在访问资源时可以增加或减少信号量。

  • 优点:可以控制同时访问共享资源的线程数量。

  • 缺点:实现复杂,容易导致资源竞争和死锁。

三:原子等待的例子:

#include <iostream>
#include <atomic>
#include <thread>
#include <chrono>

std::atomic<int> atomicValue(0);

void waitingThread() {
    std::cout << "线程正在等待原子变量的值改变...\n";
    int expected = 0;
    std::atomic_wait(&atomicValue, expected);
    std::cout << "线程恢复执行!原子变量的值变为: " << atomicValue.load() << "\n";
}

void notifyingThread() {
    std::this_thread::sleep_for(std::chrono::seconds(2)); // 模拟工作
    atomicValue.store(1);
    std::atomic_notify_one(&atomicValue);
    std::cout << "已通知等待的线程。\n";
}

int main() {
    std::thread t1(waitingThread);
    std::thread t2(notifyingThread);

    t1.join();
    t2.join();

    return 0;
}
#include <atomic>
#include <iostream>
#include <thread>

std::atomic<bool> atomicBool{};

void doTheWork(){
    std::cout << "Processing shared data." << '\n';
}

void waitingForWork(){
    std::cout << "Worker: Waiting for work." << '\n';
    atomicBool.wait(false);
    doTheWork();
    std::cout << "Work done." << '\n';
}

void setDataReady(){
    atomicBool.store(true);
    std::cout << "Sender: Data is ready."  << '\n';
    atomicBool.notify_one();
}

int main(){

    std::cout << '\n';

    std::thread t1(waitingForWork);
    std::thread t2(setDataReady);

    t1.join();
    t2.join();

    std::cout << '\n';
  
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

实时特征框架的生产实践|得物技术

一、业务背景 使用场景 推荐系统在当今的互联网应用中扮演着至关重要的角色&#xff0c;它极大地丰富了用户体验&#xff0c;帮助用户在海量信息中发现和探索他们可能感兴趣的内容。然而&#xff0c;随着数据量的激增和用户需求的日益多样化&#xff0c;传统的离线推荐系统已…

vue3+vant实现视频播放(含首次禁止进度条拖拽,视频看完后恢复,保存播放视频进度,刷新及下次进入继续播放,判断视频有无全部看完等)

1、效果图 2、 <div><videocontrolsclass"video_player"ref"videoPlayer":src"videoSrc"timeupdate"handleTimeUpdate"play"onPlay"pause"onPause"ended"onVideoEnded"></video><…

【设计模式系列】迭代器模式(七)

一、什么是迭代器模式 迭代器模式&#xff08;Iterator Pattern&#xff09;是一种行为型设计模式&#xff0c;它提供一种方法来顺序访问一个聚合对象中的各个元素&#xff0c;而不暴露其内部的表示。迭代器模式将集合的遍历过程封装在一个独立的迭代器对象中&#xff0c;这样…

硅谷甄选(10)用户管理

用户管理模块 9.1 静态搭建 主要是el-form、el-pagination <template><el-card style"height: 80px"><el-form :inline"true" class"form"><el-form-item label"用户名:"><el-input placeholder"请…

BGP路径属性与路由反射器

前言 IBGP水平分割规则用于防止AS内部产生环路&#xff0c;在很大程度上杜绝了IBGP路由产生环路的可能性&#xff0c;但是同时也带来了新的问题&#xff1a;BGP路由在AS内部只能传递一跳&#xff0c;如果建立IBGP对等体全互联模型又会加重设备的负担。 BGP 路径属性 AS_Path …

EtherCAT转ModbusTCP相关技术

EtherCAT/Ethernet/IP/Profinet/ModbusTCP协议互转工业串口网关https://item.taobao.com/item.htm?ftt&id822721028899 MS-GW15 概述 MS-GW15 是 EtherCAT 和 Modbus TCP 协议转换网关&#xff0c;为用户提供一种 PLC 扩展的集成解决方案&#xff0c;可以轻松容易将 Modbu…

如何防止U盘盗取电脑数据?

数据安全无论是对企业还是个人都至关重要。这些用户群体随时面临着数据被窃取的风险&#xff0c;而 U 盘则成为了潜在的安全隐患。如果你想要禁止电脑上使用 这类USB 存储设备&#xff0c;看完这篇文章&#xff0c;防止 U 盘盗取数据并非难事。 禁止使用usb存储设备 打开电脑上…

二叉树选择题

一 . 相关性质 1 &#xff09; 对于任何一棵二叉树 &#xff0c; 如果度数为 0 ---> 其叶子结点个数为 , 度数为2的分支结点个数为 &#xff0c;则有 2 ) 树的结点总数 n 3 ) 有 n 个结点 &#xff0c; 则边数为 n - 1 4 ) 满二叉树 &#xff1a;&#xff08;k 层) 结…

【急救】——心肺复苏和AED使用以及海姆立克法

【急救】——心肺复苏和AED使用以及海姆立克法 法律保障成人CPR实施步骤1.确保现场环境安全和自身安全2.呼唤确认救助对象状况&#xff0c;通过轻拍肩膀&#xff0c;触摸腹部数10个数识别呼吸***3.呼喊求助***4.找到按压位置5.按照标准要求按压什么时候停止CPR(心肺复苏&#x…

【小白学机器学习28】 统计学脉络+ 总体+ 随机抽样方法

目录 参考书&#xff0c;学习书 0 统计学知识大致脉络 1 个体---抽样---整体 1.1 关于个体---抽样---整体&#xff0c;这个三段式关系 1.2 要明白&#xff0c;自然界的整体/母体是不可能被全部认识的 1.2.1 不要较真&#xff0c;如果是人为定义的一个整体&#xff0c;是可…

Qgis 开发初级 《ToolBox》

Qgis 有个ToolBox 的&#xff0c;在Processing->ToolBox 菜单里面&#xff0c;界面如下。 理论上Qgis这里面的工具都是可以用脚本或者C 代码调用的。界面以Vector overlay 为例子简单介绍下使用方式。Vector overlay 的意思是矢量叠置分析&#xff0c;和arcgis软件类似的。点…

用图说明 CPU、MCU、MPU、SoC 的区别

CPU CPU 负责执行构成计算机程序的指令&#xff0c;执行这些指令所指定的算术、逻辑、控制和输入/输出&#xff08;I/O&#xff09;操作。 MCU (microcontroller unit) 不同的 MCU 架构如下&#xff0c;注意这里的 MPU 表示 memory protection unit MPU (microprocessor un…

HCIA(ACL)

第七节 ACL&#xff1a;访问控制列表 访问控制----在路由器的入或者出的接口上&#xff0c;匹配流量&#xff0c;之后产生动作---允许或拒绝 定义感兴趣流量-----帮助其他软件抓流量 匹配规则&#xff1a; 至上而下&#xff0c;逐一匹配&#xff0c;上调匹配按照上条执行…

冒泡排序和二分查找--go

冒泡排序的逻辑 二分查找的逻辑 func bubbleSort(arr *[5]int){//冒泡排序fmt.Println(*arr)temp : 0for j : len(*arr); j > 0; j-- {for i : 0; i < j-1; i {temp (*arr)[i]if((*arr)[i] > (*arr)[i1]){(*arr)[i] (*arr)[i1](*arr)[i1] temp}}} }func binaryF…

【工具分享】Pylocky勒索病毒解密工具

前言 PyLocky勒索软件首次出现在2018年&#xff0c;以模仿著名的Locky勒索软件而得名。与Locky无实际关联&#xff0c;PyLocky是用Python编写的&#xff0c;并通过PyInstaller打包成可执行文件&#xff0c;使其更难被检测。PyLocky通常通过网络钓鱼邮件传播&#xff0c;邮件伪…

SAP-FICO 月结流程

一、财务月结 1、资产会计-固定资产折旧计提AFAB 正式运行之前&#xff0c;先测试运行&#xff0c;没有问题就正式运行 可以看到&#xff0c;没有错误 因为正式系统的资产会过于庞大&#xff0c;一般都是后台运行资产的折旧 点击程序--后台执行 AFBP查看运行日志&#xff0c…

vscode | 开发神器vscode快捷键删除和恢复

目录 快捷键不好使了删除快捷键恢复删除的快捷键 在vscode使用的过程中&#xff0c;随着我们自身需求的不断变化&#xff0c;安装的插件将会持续增长&#xff0c;那么随之而来的就会带来一个问题&#xff1a;插件的快捷键重复。快捷键重复导致的问题就是快捷键不好使了&#xf…

ETLCloud怎么样?深度解析其在数据管理中的表现

在BI或数据大屏等数据分析工具中&#xff0c;经常需要从多个业务系统中提取原始数据&#xff0c;然后对数据进行清洗、处理&#xff0c;以获取高质量、有效且干净的数据以供后续的BI进行数据统计和分析使用&#xff0c;从高质量的实现企业数据的价值变现。 然而&#xff0c;在…

0xGame 2024 [Week 4] Jenkins

1.前言 由于好久没做web题了&#xff0c;所以今天来尝试来做一波web题&#xff0c;仅供刷题记录。 2.题目 这个给的提示对于小白来说实在是友好的过劲。 3.分析 上网搜到一个关于Jenkins的历史漏洞&#xff0c;下面链接可供参考 https://blog.csdn.net/2301_80127209/arti…