C++11 线程同步接口std::condition_variable和std::future的简单使用

news/2025/1/12 18:55:38/文章来源:https://www.cnblogs.com/stephen2023/p/18416810

std::condition_variable

条件变量std::condition_variable有wait和notify接口用于线程间的同步。如下图所示,Thread 2阻塞在wait接口,Thread 1通过notify接口通知Thread 2继续执行。

con_variable_result

具体参见示例代码:

#include<iostream>
#include<mutex>
#include<thread>
#include<queue>
std::mutex mt;
std::queue<int> data;
std::condition_variable cv;
auto start=std::chrono::high_resolution_clock::now();void logCurrentTime()
{auto end = std::chrono::high_resolution_clock::now();auto elapsed = std::chrono::duration_cast<std::chrono::milliseconds>(end - start).count();std::cout << elapsed << ":";
}
void prepare_data()
{	logCurrentTime();std::cout << "this is " << __FUNCTION__ << " thread:" << std::this_thread::get_id() << std::endl;for (int i = 0; i < 10; i++){data.push(i);logCurrentTime();std::cout << "data OK:" << i << std::endl;}//start to notify consume_data thread data is OK!cv.notify_one();
}void consume_data()
{logCurrentTime();std::cout << "this is: " << __FUNCTION__ << " thread:" << std::this_thread::get_id() << std::endl;std::unique_lock<std::mutex> lk(mt);//wait first for notificationcv.wait(lk);  //it must accept a unique_lock parameter to waitwhile (!data.empty()){logCurrentTime();std::cout << "data consumed: " << data.front() << std::endl;data.pop();}
}int main()
{std::thread t2(consume_data);//wait for a while to wait first then prepare data,otherwise stuck on waitstd::this_thread::sleep_for(std::chrono::milliseconds(10));std::thread t1(prepare_data);t1.join();t2.join();return 0;
}

输出结果

con_variable_result

分析

主线程中另启两个线程,分别执行consume_data和prepare_data,其中consume_data要先执行,以保证先等待再通知,否则若先通知再等待就死锁了。首先consume_data线程在从wait 处阻塞等待。后prepare_data线程中依次向队列写入0-10,写完之后通过notify_one 通知consume_data线程解除阻塞,依次读取0-10。

std::future

std::future与std::async配合异步执行代码,再通过wait或get接口阻塞当前线程等待结果。如下图所示,Thread 2中future接口的get或wait接口会阻塞当前线程,std::async异步开启的新线程Thread1执行结束后,将结果存于std::future后通知Thread 1获取结果后继续执行.

con_variable_result

具体参见如下代码:

#include <iostream>
#include <future>
#include<thread>int test()
{std::cout << "this is " << __FUNCTION__ << " thread:" << std::this_thread::get_id() << std::endl;;std::this_thread::sleep_for(std::chrono::microseconds(1000));return 10;
}
int main()
{std::cout << "this is " <<__FUNCTION__<<" thread:" << std::this_thread::get_id() << std::endl;;//this will lanuch on another threadstd::future<int> result = std::async(test);std::cout << "After lanuch a thread: "<< std::this_thread::get_id() << std::endl;//block the thread and wait for the resultstd::cout << "result is: " <<result.get()<< std::endl;std::cout << "After get result "<< std::endl;return 0;
}

输出结果

运行结果

分析

主程序中调用std::async异步调用test函数,可以看到main函数的线程ID 27428与test函数执行的线程ID 9704并不一样,说明std::async另起了一个新的线程。在test线程中,先sleep 1000ms,所以可以看到"After lanuch a thread:"先输出,说明主线程异步执行,不受子线程影响。而"After get result "最后输出,说明get()方法会阻塞主线程,直到获取结果。

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

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

相关文章

macOS Sequoia 15.0 (24A335) 正式版发布,ISO、IPSW、PKG 下载

macOS Sequoia 15.0 (24A335) 正式版发布,ISO、IPSW、PKG 下载macOS Sequoia 15.0 (24A335) 正式版发布,ISO、IPSW、PKG 下载 iPhone 镜像、Safari 浏览器重大更新、备受瞩目的游戏和 Apple Intelligence 等众多全新功能令 Mac 使用体验再升级 请访问原文链接:https://sysin…

Go runtime 调度器精讲(十一):总览全局

原创文章,欢迎转载,转载请注明出处,谢谢。0. 前言 前面用了十讲介绍了 Go runtime 调度器,这一讲结合一些图在总览下 Go runtime 调度器。 1. 状态转换图 首先是 Goroutine 的状态转换图:大部分转移路径前面几讲也介绍过,这里就不继续介绍了(下同)。 接着是 P 的状态转移…

macOS Sonoma 14.7 (23H124) 正式版发布,ISO、IPSW、PKG 下载

macOS Sonoma 14.7 (23H124) 正式版发布,ISO、IPSW、PKG 下载macOS Sonoma 14.7 (23H124) 正式版发布,ISO、IPSW、PKG 下载 2024 年 9 月 17 日凌晨 1 点,Tim Cook 领导的 Apple 今天发布了 macOS 15 Sequoia 正式版,iPhone 镜像、密码应用程序、窗口平铺更新等带来全新体验…

CentOS 设置静态地址

配置文件路径: /etc/sysconfig/network-scripts/ 在ifcfg-ensxxx配置文件里做出如下修改:在最后添加以下配置 IPADDR=192.168.10.10 #静态IP GATEWAY=192.168.10.1 #默认网关 NETMASK=255.255.255.0 #子网掩码 DNS1=192.168.10.1 #DNS 配置

day06 数据类型:指针、切片、字典

day06 数据类型 Go语言中常见的数据类型有很多,例如:整型,用于表示整数。 浮点型,用于表示小数。 布尔型,用于表示真/假。 字符串,用于表示文本信息。 数组,用于表示多个数据(数据集合) 指针,用于表示内存地址的类型。 切片,用于表示多个数据(数据集合) 字典,用于…

day05 数据类型

day05 数据类型写程序 等价于 写作文数据类型,其实就是各种各样类型的数据。Go语言中常见的数据类型有挺多,例如:整型,用于表示整数。 浮点型,用于表示小数。 布尔型,用于表示真/假。 字符串,用于表示文本信息。 数组,用于表示多个数据(数据集合) 指针,用于表示内存…

前端系列一:HTML

HTML:HyperText Markup Language,超文本标记语言。Author: ACatSmiling Since: 2024-09-10HTML 4 HTML:HyperText Markup Language,超文本标记语言。超文本的含义:是一种组织信息的方式,通过超链接将不同空间的文字、图片等各种信息组织在一起,能从当前阅读的内容,跳转…

day03 基础知识

day03 基础知识 今日概要switch case语句,条件判断。 for循环语句,循环。 goto语法,不太建议使用。 字符串格式化,“拼接”数据。 运算符1.switch语句 package mainfunc main() {// 表达式/*switch 1 + 1 {case 1:fmt.Println("等于1")case 2:fmt.Println("…

day02 快速上手

day02 快速上手 今日概要初识包管理,知道项目中文件和文件、文件和文件夹之间关系。输出,写代码,在go编译器运行时会在屏幕显示内容。初识数据类型整型,数字。例如:1、2、3、4 字符串,表示文本信息。例如:“如家” "锦江之星" 布尔类型,真假。例如: 1>2…

在研究yum源的时候发现的一个小工具

在确保联网的情况下 在终端输入以下命令: bash <(curl -sSL https://gitee.com/SuperManito/LinuxMirrors/raw/main/ChangeMirrors.sh)

day01 环境搭建

day01 环境搭建 Go 和 C语言、C++、Python、Java 等一样都是编程语言。学习任何一门编程语言本质上都分3步走:第一步:安装 解释器 或 编译器。 第二步:学相关编程语言语法,然后写代码。 第三步:用已安装解释器 或 编译器 去运行自己写的代码,这样代码就会去完成我们编写的…

day01 GO环境搭建

day01 环境搭建 Go 和 C语言、C++、Python、Java 等一样都是编程语言。学习任何一门编程语言本质上都分3步走:第一步:安装 解释器 或 编译器。 第二步:学相关编程语言语法,然后写代码。 第三步:用已安装解释器 或 编译器 去运行自己写的代码,这样代码就会去完成我们编写的…