c++进阶篇——初窥多线程(五) 条件变量与信号量

条件变量

什么是条件变量

条件变量是线程间同步的一种机制,它允许一个或多个线程在某些条件满足时被唤醒,从而继续执行。条件变量通常与互斥锁一起使用,以确保线程在访问共享资源时不会发生竞争条件,常见的条件变量主要有以下几种:

  • condition-variable:提供与 std::unique_lock关联的条件变量
  • condition_variable_any:提供与任何锁类型关联的条件变量

两者的主要区别在于,condition_variable_any可以凭借unique_lock来与任何类型的锁一起使用,而condition_variable只能与std::mutex一起使用。

条件变量的基本操作

条件变量的操作主要包括两个:

  • wait:等待条件变量,直到条件满足
  • notify_one/notify_all:通知等待条件变量的线程

关于这两个操作的常见api主要有下面几种:

  • wait:等待条件变量,直到条件满足
  • wait_for:等待条件变量,直到条件满足或超时
  • wait_until:等待条件变量,直到条件满足或指定的时间点
  • notify_one:通知一个等待条件变量的线程
  • notify_all:通知所有等待条件变量的线程

基于条件变量的生产者消费者模型

下面我们来看一下如何使用上面所说的条件变量来实现一个生产者消费者模型(考虑到两个条件变量的用法基本一致,我们只以condition_variable为例):

//test.h
#include <iostream>
#include <condition_variable>
#include <thread>
#include <chrono>
#include <mutex>
#include <list>
#include <functional>using namespace std;
#define MAX_SIZE 100 // 任务队列的最大长度template <typename T>
class SyncQueue
{
private:mutex m_mutex;                // 互斥锁condition_variable non_empty; // 判断当前生产者消费者模型中的任务队列是否为空condition_variable non_full;  // 判断当前生产者消费者模型中的任务队列是否为满list<T> m_queue;              // 任务队列int m_maxsize;                // 任务队列的最大长度
public:SyncQueue() : m_maxsize(MAX_SIZE) {} // 构造函数void put(T t)                        // 生产函数{unique_lock<mutex> lock(m_mutex); // 这里通过unique_lock来获取互斥锁,保证线程安全non_full.wait(lock, [this](){ return m_queue.size() != m_maxsize; }); // 等待任务队列不满m_queue.push_back(t);cout << t << " 被生产" << endl; non_empty.notify_one(); // 唤醒一个消费者线程}T take() // 消费函数{unique_lock<mutex> lock(m_mutex); // 这里通过unique_lock来获取互斥锁,保证线程安全non_empty.wait(lock, [this](){ return m_queue.size() > 0; }); // 等待任务队列不为空T t = m_queue.front();m_queue.pop_front();non_full.notify_one(); // 唤醒一个生产者线程cout << t << " 被消费" << endl; return t;}bool empty(){unique_lock<mutex> locker(m_mutex);return m_queue.empty();}bool full(){unique_lock<mutex> locker(m_mutex);return m_queue.size() == m_maxsize;}int size(){unique_lock<mutex> locker(m_mutex);return m_queue.size();}~SyncQueue() {} // 析构函数
};//test1.cpp
#include "./include/test.h"
int main()
{SyncQueue<int> q;auto produce=std::bind(&SyncQueue<int>::put,&q,std::placeholders::_1);auto consume=std::bind(&SyncQueue<int>::take,&q);thread t1[3],t2[3];for(int i=0;i<3;i++){t1[i]=thread(produce,i+100);t2[i]=thread(consume);}for(int i=0;i<3;i++){t1[i].join();t2[i].join();}return 0;
}

同时附上cmake文件(仅供参考):

# 设置打印编译信息
set(CMAKE_VERBOSE_MAKEFILE ON)# 设置cmake编译器最低版本
cmake_minimum_required (VERSION 3.10)project (learn_thread)#头文件搜素路径
include_directories(${PROJECT_SOURCE_DIR}/include)# 添加可执行文件
add_executable(demo1 test1.cpp)# 添加链接库
link_libraries(demo1 PRIVATE Threads::Threads)# 添加编译选项
add_compile_options(-c -g)

信号量

在多进程编程中我们就对信号量做了一定的介绍,在cpp20中,并发库对信号量做了封装实现,让我们对信号量有更方便的操作。

信号量的种类

cpp11中信号量分为两种,一种为计数信号量,一种为二值信号量。

  • 计数信号量:计数信号量可以用于控制多个线程对共享资源的访问,当计数信号量大于0时,线程可以访问共享资源,当计数信号量等于0时,线程需要等待信号量释放。
  • 二值信号量:二值信号量只能用于控制两个线程对共享资源的访问,当二值信号量等于1时,线程可以访问共享资源,当二值信号量等于0时,线程需要等待信号量释放。

其实本质上来说,二值信号量实现的是一个类似于互斥量的功能,但是二值信号量起到的更多的是一个信号的作用,他是通知线程之间的事件,比如线程是应该执行还是阻塞,而互斥量起到的更多的是一个锁的作用,他是保护共享资源不被多个线程同时访问。

而计数信号量则可以用于控制多个线程对共享资源的访问,当计数信号量大于0时,线程可以访问共享资源,当计数信号量等于0时,线程需要等待信号量释放。

信号量的api

虽然信号量的之类有二值和计数两种,但是他们的api都是一样的,只是初始化的参数不同,首先我们来看一下它们共有的api;

  • release():释放信号量,将信号量的计数加1。
  • acquire():获取信号量,将信号量的计数减1,如果计数小于0,则线程需要等待信号量释放。
  • try_acquire():尝试获取信号量,将信号量的计数减1,如果计数小于0,则返回false,否则返回true。
  • try_acquire_for():尝试获取信号量,将信号量的计数减1,如果计数小于0,则等待一段时间,如果等待时间到了,则返回false,否则返回true。
  • try_acquire_until():尝试获取信号量,将信号量的计数减1,如果计数小于0,则等待直到指定的时间,如果等待时间到了,则返回false,否则返回true。

信号量的初始化

信号量的初始化有两种方式,一种是通过构造函数,一种是通过赋值操作符。

  • 构造函数:构造函数需要传入一个整数,表示信号量的初始计数。

信号量的使用

首先是binary_semaphore,我们将演示如何使用binary_semaphore来控制两个线程对共享资源的访问。

#include <iostream>
#include <thread>
#include <semaphore>
#include <chrono>
#include <mutex>
#include <random>std::binary_semaphore binary_lock(1); // 初始状态为有信号
int count=0; // 共享变量void add()
{for(int i=0;i<100000;i++){binary_lock.acquire(); // 获取信号量,如果信号量为0,则阻塞当前线程(信号量的P操作)count++;binary_lock.release(); // 释放信号量,将信号量加1(信号量的V操作);}
}int main()
{std::thread t1(add);std::thread t2(add);t1.join();t2.join();std::cout<<"count="<<count<<std::endl;return 0;
}

然后是count_semaphore,我们将演示如何用它配合std::mutex来控制对共享资源的访问,这里我们以生产者消费者模型为例:

#include <iostream>
#include <queue>
#include <thread>
#include <chrono>
#include <mutex>
#include <condition_variable>
#include <random>const int BUFFER_SIZE = 10;
std::queue<int> buffer;
std::mutex buffer_mutex;
std::condition_variable cond_var;
bool stop_threads = false;// 生产者和消费者的逻辑保持不变,但是需要监听stop_threads标志。void producer(int id) {int count = 0;std::random_device rd;std::mt19937 gen(rd());std::uniform_int_distribution<> dis(1000, 5000);while (!stop_threads) {std::unique_lock<std::mutex> lk(buffer_mutex);cond_var.wait(lk, [] { return buffer.size() < BUFFER_SIZE; }); // 等待缓冲区非满buffer.push(count);std::cout << "Produced: " << count << std::endl;count++;lk.unlock();cond_var.notify_one(); // 通知消费者std::this_thread::sleep_for(std::chrono::milliseconds(dis(gen)));}
}void consumer(int id) {//定义随机数生成器std::random_device rd;std::mt19937 gen(rd());std::uniform_int_distribution<> dis(1000, 5000);while (!stop_threads) {std::unique_lock<std::mutex> lk(buffer_mutex);cond_var.wait(lk, [] { return !buffer.empty(); }); // 等待缓冲区非空int value = buffer.front();buffer.pop();std::cout << "Consumed: " << value << std::endl;lk.unlock();cond_var.notify_one(); // 通知生产者std::this_thread::sleep_for(std::chrono::milliseconds(dis(gen)));  // 模拟消费时间}
}int main() {std::thread prod(producer, 0);std::thread cons(consumer, 0);prod.join();cons.join();// 设置停止标志并唤醒所有等待中的线程stop_threads = true;cond_var.notify_all();return 0;
}

备注:

  • 由于上面的有关信号量的信号量封装类是cpp20的产物,所以我们编译的时候药品注意编译器版本,编译命令可以参考:
g++ -std=c++20 -o test test2.cpp
  • 在信号量的初始化函数中其实STL标准库中只有
std::counting_semaphore<LeastMaxValue>::counting_semaphore

在源码中关于binary_semaphore的初始化函数是这样的:

using binary_semaphore = std::counting_semaphore<1>;

所以两种信号量本质上用法没有什么太大区别,只不过binary_semaphorecount_semphore的一个特例,我们在使用时

std::binary_semaphore binary_lock(1);

也不过是初始化二值信号量的状态。

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

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

相关文章

缓存数据减轻服务器压力

问题:不是所有的数据都需要请求后端的 不是所有的数据都需要请求后端的,有些数据是重复的、可以复用的解决方案:缓存 实现思路:每一个分类为一个key,一个可以下面可以有很多菜品 前端是按照分类查询的,所以我们需要通过分类来缓存缓存代码 /*** 根据分类id查询菜品** @pa…

Linux中的进程间通信之共享内存

共享内存 共享内存示意图 共享内存数据结构 struct shmid_ds {struct ipc_perm shm_perm; /* operation perms */int shm_segsz; /* size of segment (bytes) */__kernel_time_t shm_atime; /* last attach time */__kernel_time_t shm_dtime; /* last detach time */__kerne…

[Linux] Linux 初识进程地址空间 (进程地址空间第一弹)

标题&#xff1a;[Linux] Linux初识进程地址空间 个人主页水墨不写bug &#xff08;图片来源于AI&#xff09; 目录 一、什么是进程地址空间 二、为什么父子进程相同地址的变量的值不同 三、初识虚拟地址、页表 一、什么是进程地址空间 其实&#xff0c;在很久之前&#xf…

【S32K3 RTD MCAL 篇1】 K344 KEY 控制 EMIOS PWM

【S32K3 RTD MCAL 篇1】 K344 KEY 控制 EMIOS PWM 一&#xff0c;文档简介二&#xff0c; 功能实现2.1 软硬件平台2.2 软件控制流程2.3 资源分配概览2.4 EB 配置2.4.1 Dio module2.4.2 Icu module2.4.4 Mcu module2.4.5 Platform module2.4.6 Port module2.4.7 Pwm module 2.5 …

STM32+ADC+扫描模式

1 ADC简介 1 ADC(模拟到数字量的桥梁) 2 DAC(数字量到模拟的桥梁)&#xff0c;例如&#xff1a;PWM&#xff08;只有完全导通和断开的状态&#xff0c;无功率损耗的状态&#xff09; DAC主要用于波形生成&#xff08;信号发生器和音频解码器&#xff09; 3 模拟看门狗自动监…

Oracle架构之数据库备份和RAC介绍

文章目录 1 数据库备份1.1 数据库备份分类1.1.1 逻辑备份与物理备份1.1.2 完全备份/差异备份/增量备份 1.2 Oracle 逻辑备份1.2.1 EXP/IMP1.2.1.1 EXP导出1.2.1.2 EXP关键字说明1.2.1.3 导入1.2.1.4 IMP关键字说明 1.2.2 EXPDP/IMPDP1.2.2.1 数据泵介绍1.2.2.2 数据泵的使用 1.…

通过 Groovy 实现业务逻辑的动态变更

Groovy 1、需求的提出2、为什么是Groovy3、设计参考1_引入Maven依赖2_GroovyEngineUtils工具类3_GroovyScriptVar类4_脚本规则表设计5_对应的实体类6_数据库访问层7_GroovyExecService通用接口 4、测试5、其他的注意事项6、总结 1、需求的提出 在我们日常的开发过程中&#xf…

嵌入式知识点复习(一)

国庆倒数第二天&#xff0c;进行嵌入式课堂测试的复习&#xff1a; 第一章 绪论 1.1 嵌入式系统的概念 嵌入式系统定义 嵌入式系统定位 嵌入式系统形式 嵌入式系统三要素 嵌入式系统与桌面通用系统的区别 1.2 嵌入式系统的发展历程 微处理器的演进历史 单片机的演进历史 …

【易社保-注册安全分析报告】

前言 由于网站注册入口容易被黑客攻击&#xff0c;存在如下安全问题&#xff1a; 1. 暴力破解密码&#xff0c;造成用户信息泄露 2. 短信盗刷的安全问题&#xff0c;影响业务及导致用户投诉 3. 带来经济损失&#xff0c;尤其是后付费客户&#xff0c;风险巨大&#xff0c;造…

【Python】数据可视化之聚类图

目录 clustermap 主要参数 参考实现 clustermap sns.clustermap是Seaborn库中用于创建聚类热图的函数&#xff0c;该函数能够将数据集中的样本按照相似性进行聚类&#xff0c;并将聚类结果以矩阵的形式展示出来。 sns.clustermap主要用于绘制聚类热图&#xff0c;该热图通…

用manim实现Gram-Schmidt正交化过程

在线性代数中&#xff0c;正交基有许多美丽的性质。例如&#xff0c;由正交列向量组成的矩阵(又称正交矩阵)可以通过矩阵的转置很容易地进行反转。此外&#xff0c;例如&#xff1a;在由彼此正交的向量张成的子空间上投影向量也更容易。Gram-Schmidt过程是一个重要的算法&#…

Python Tips6 基于数据库和钉钉机器人的通知

说明 起因是我第一版quant程序的短信通知失效了。最初认为短信是比较即时且比较醒目的通知方式&#xff0c;现在看来完全不行。 列举三个主要问题&#xff1a; 1 延时。在早先还能收到消息的时候&#xff0c;迟滞就很严重&#xff0c;几分钟都算短的。2 完全丢失。我手机没有…

AI 时代:产品经理不“AI”就出局?

即便你没想去做“AI 产品经理”&#xff0c;那你也不能成为一个不会用 AI 的产品经理。 产品经理肯定是所有互联网从业者中&#xff0c;最先捕捉到 AI 趋势的岗位。 但只知道 AI、关注 AI 还不够&#xff0c;仔细审视一下&#xff1a;你自己的工作&#xff0c;被 AI 提效了么…

《Windows PE》4.1导入表

导入表顾名思义&#xff0c;就是记录外部导入函数信息的表。这些信息包括外部导入函数的序号、名称、地址和所属的DLL动态链接库的名称。Windows程序中使用的所有API接口函数都是从系统DLL中调用的。当然也可能是自定义的DLL动态链接库。对于调用方&#xff0c;我们称之为导入函…

安防监控/视频系统EasyCVR视频汇聚平台如何过滤134段的告警通道?

视频汇聚/集中存储EasyCVR安防监控视频系统采用先进的网络传输技术&#xff0c;支持高清视频的接入和传输&#xff0c;能够满足大规模、高并发的远程监控需求。平台支持国标GB/T 28181协议、部标JT808、GA/T 1400协议、RTMP、RTSP/Onvif协议、海康Ehome、海康SDK、大华SDK、华为…

STM32定时器(TIM)

目录 一、概述 二、定时器的类型 三、时序 四、定时器中断基本结构 五、定时器定时中断代码 六、定时器外部时钟代码 一、概述 TIM(Timer)定时器 定时器可以对输入的时钟进行计数&#xff0c;并在计数值达到设定值时触发中断16位计数器、预分频器、自动重装寄存器的时基…

力扣刷题 | 两数之和

目前主要分为三个专栏&#xff0c;后续还会添加&#xff1a; 专栏如下&#xff1a; C语言刷题解析 C语言系列文章 我的成长经历 感谢阅读&#xff01; 初来乍到&#xff0c;如有错误请指出&#xff0c;感谢&#xff01; 给定一个整数数组 nums 和…

网站建设完成后,切勿让公司官网成为摆设

在当今这个数字化时代&#xff0c;公司官网已经成为企业展示形象、传递信息、吸引客户的重要平台。然而&#xff0c;许多企业在网站建设完成后&#xff0c;往往忽视了对官网的持续运营和维护&#xff0c;导致官网逐渐沦为摆设&#xff0c;无法发挥其应有的作用。为了确保公司官…

15分钟学 Python 第40天:Python 爬虫入门(六)第一篇

Day40 &#xff1a;Python 爬取豆瓣网前一百的电影信息 1. 项目背景 在这个项目中&#xff0c;我们将学习如何利用 Python 爬虫技术从豆瓣网抓取前一百部电影的信息。通过这一练习&#xff0c;您将掌握网页抓取的基本流程&#xff0c;包括发送请求、解析HTML、存储数据等核心…

jvisualvm学习

系列文章目录 JavaSE基础知识、数据类型学习万年历项目代码逻辑训练习题代码逻辑训练习题方法、数组学习图书管理系统项目面向对象编程&#xff1a;封装、继承、多态学习封装继承多态习题常用类、包装类、异常处理机制学习集合学习IO流、多线程学习仓库管理系统JavaSE项目员工…