【C++学习(36)】C++20的co_await 的不同使用方式和特性

这三个代码片段的目的是展示 C++ 中 协程(Coroutines)的不同使用方式和特性。每个代码展示了不同类型的协程调度和挂起/恢复机制。

1. 第一个代码:使用 std::experimental::coroutine_handle 手动控制协程恢复

关键点:
  • 手动控制协程的恢复myCoroutine.resume() 手动恢复协程执行。
  • 使用 coroutine_handle 保存协程myCoroutine 是一个全局变量,存储了当前协程的句柄,以便后续在 myTask() 中恢复协程。
解析:

该示例中展示了一个手动控制协程的例子:

  • myTask 中的 myCoroutine.resume() 表示手动恢复协程,调用它启动协程的执行。
  • asyncFunc 中使用 co_await MyAwaitable{} 来挂起当前协程,并且通过 await_suspend 将协程句柄保存(即 myCoroutine = handle),以便后续恢复。
  • 这种方式展示了如何手动控制协程的挂起与恢复。
主要学习内容:
  • 如何通过 std::experimental::coroutine_handle 控制协程的恢复。
  • co_awaitawait_suspend 的基本用法。
#include <iostream>
#include <experimental/coroutine>std::experimental::coroutine_handle<> myCoroutine;void myTask() {std::cout << "Starting coroutine..." << std::endl;myCoroutine.resume(); // 启动协程std::cout << "Resuming execution..." << std::endl;
}struct MyAwaitable {bool await_ready() const { return false; }void await_suspend(std::experimental::coroutine_handle<> handle) const {myCoroutine = handle; // 将当前协程保存起来}void await_resume() const {}
};MyAwaitable asyncFunc() {std::cout << "Suspending execution..." << std::endl;co_await MyAwaitable{}; // 挂起协程,等待恢复std::cout << "Resumed execution..." << std::endl;
}int main() {myTask();asyncFunc().await_resume();return 0;
}

2. 第二个代码:使用 std::experimental::suspend_always 自动恢复协程

关键点:
  • 协程自动恢复:使用 handle.resume()await_suspend 中自动恢复协程。
  • 使用 suspend_alwaysstd::experimental::suspend_always 是一种简单的挂起机制,协程会在执行到 co_await 时自动挂起,直到 resume() 被调用。
解析:

在此示例中:

  • 协程 asyncFunc 在执行到 co_await MyAwaitable{} 时会被挂起。
  • MyAwaitable 结构体的 await_suspend 函数通过调用 handle.resume() 自动恢复协程的执行。
  • await_ready 返回 false,表示协程总是会被挂起。
主要学习内容:
  • 如何使用 suspend_always 来让协程在执行时自动挂起,直到外部调用 resume() 恢复。
  • 使用 handle.resume() 恢复协程执行的基本用法。
#include <iostream>
#include <experimental/coroutine>struct MyAwaitable {bool await_ready() const { return false; }void await_suspend(std::experimental::coroutine_handle<> handle) const {std::cout << "Suspending coroutine..." << std::endl;handle.resume(); // 恢复协程的执行}void await_resume() const {}
};std::experimental::suspend_always asyncFunc() {std::cout << "Starting coroutine..." << std::endl;co_await MyAwaitable{}; // 使用 co_await 挂起协程std::cout << "Resuming execution..." << std::endl;
}int main() {auto coro = asyncFunc();coro.resume(); // 启动协程return 0;
}

3. 第三个代码:协程与异步操作结合(例如异步请求)

关键点:
  • 与异步操作结合:结合了 C++ 的 std::future 和协程,通过 await_suspend 在异步操作完成后恢复协程。
  • 模拟异步请求:通过 std::promisestd::future 模拟了一个异步请求,协程在等待结果时被挂起,异步请求完成后恢复协程。
解析:

该代码片段展示了如何使用协程与异步操作(如线程和 std::future)配合:

  • makeAsyncRequest 函数创建了一个异步操作(通过 std::thread 模拟延迟的任务),返回一个 Awaitable 对象。
  • Awaitable 中,await_ready 判断异步结果是否已经准备好,如果没有准备好,await_suspend 会将当前协程挂起,直到异步操作完成。
  • std::future 中的结果准备好时,协程会被恢复执行。
主要学习内容:
  • 协程与异步操作(如 std::future)的结合使用。
  • 使用 std::promisestd::future 模拟异步操作,并结合协程来实现异步任务的等待与恢复。
#include <iostream>
#include <future>
#include <experimental/coroutine>struct Awaitable {std::future<int> fut;bool await_ready() {return fut.wait_for(std::chrono::seconds(0)) == std::future_status::ready;}void await_suspend(std::experimental::coroutine_handle<> handle) {fut.then([handle](std::future<int> f) mutable {// 异步操作完成后恢复协程执行handle.resume();});}int await_resume() {return fut.get();}
};Awaitable makeAsyncRequest() {std::promise<int> p;auto fut = p.get_future();// 模拟异步操作std::thread([&p]() mutable {std::this_thread::sleep_for(std::chrono::seconds(2));p.set_value(42);}).detach();return Awaitable{std::move(fut)};
}std::experimental::suspend_always task() {auto result = co_await makeAsyncRequest(); // 等待异步操作完成,并获取结果std::cout << "Result: " << result << std::endl;
}int main() {auto coro = task();coro.resume(); // 启动协程return 0;
}

总结:

  1. 第一个代码展示了手动控制协程的恢复和执行,重点在于如何使用 std::experimental::coroutine_handle 来手动恢复协程。
  2. 第二个代码展示了如何使用 std::experimental::suspend_always 来让协程自动挂起,并通过 handle.resume() 来恢复协程。
  3. 第三个代码展示了如何将协程与异步操作(如 std::future)结合使用,模拟了异步请求并在请求完成后恢复协程的执行。

这些例子帮助理解 C++20 协程的核心机制,尤其是如何通过 co_awaitawait_suspendstd::future 等方式进行协程挂起与恢复。

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

谷歌Gemini发布iOS版App,live语音聊天免费用!

大家好&#xff0c;我是木易&#xff0c;一个持续关注AI领域的互联网技术产品经理&#xff0c;国内Top2本科&#xff0c;美国Top10 CS研究生&#xff0c;MBA。我坚信AI是普通人变强的“外挂”&#xff0c;专注于分享AI全维度知识&#xff0c;包括但不限于AI科普&#xff0c;AI工…

MYSQL SQL优化总结【快速理解】

1、优化insert操作 批量插入&#xff0c;防止大量与数据库进行访问 手动控制事务&#xff0c;减少事务的频繁开启和提交。 主键顺序插入 2、优化主键 主键优化的点就是避免主键过长&#xff0c;因为如果有二级索引&#xff0c;叶子节点存储的数据时间上是主键&#xff0c;如果主…

【Hive】【HiveQL】【大数据技术基础】 实验四 HBase shell命令实验

实验四&#xff1a;熟悉常用的HBase操作 实验概览 在本次实验中&#xff0c;我们将深入探索HBase在Hadoop生态系统中的角色&#xff0c;并熟练掌握常用的HBase Shell命令和Java API操作。通过这些实践&#xff0c;我们能够更好地理解HBase的工作原理以及如何在实际项目中应用。…

解决渠道问题的高效控价方法

当品牌销售渠道增多、涉及销售店铺量上升且品牌期望持续稳定发展时&#xff0c;就应着手处理控价事宜。控价有助于稳定品牌价值、吸引经销商加入以及利于品牌口碑传播。 控价应包含的渠道 随着电商平台发展&#xff0c;品牌销售渠道日益丰富&#xff0c;除线下传播渠道外&…

随机数

目录 一、传统方式&#xff1a;std::rand 和 std::srand 使用方法&#xff1a; 优缺点&#xff1a; 二、现代方式&#xff1a; 库&#xff08;推荐&#xff09; 1. 随机整数 2. 随机浮点数 3. 布尔值 4. 字符 5. 正态分布&#xff08;高斯分布&#xff09; 6. 离散分…

生物信息入门软件安装(保姆级教程)

写在开头 大四期间&#xff0c;选修了一门智慧医疗的课程&#xff0c;期末考核为25分钟有关智慧医疗方面的汇报。一次偶然的课程汇报让我接触到了生物信息&#xff0c;也产生了浓厚的兴趣&#xff0c;同时加入了老师的研究生小组&#xff0c;开启了这段生物信息学习的旅途。至此…

数据智能新纪元:向量数据库驱动AI大模型创新

数据智能新纪元&#xff1a;向量数据库驱动AI大模型创新 前言向量数据库与AI大模型的关系以及发展现状向量数据库的技术创新与落地实践向量数据库的未来趋势与产业机遇 前言 最近和一位搞AI创业的朋友聊天&#xff0c;他说了句意味深长的话&#xff1a;“当所有人都在谈论大模型…

2024 Visual Studio Code的下载与安装

目录 一、Windows系统下载与安装二、macOS系统下载与安装三、Linux系统下载与安装四、启动与配置VS Code 以下是2024年Visual Studio Code&#xff08;简称VS Code&#xff09;的下载与安装步骤&#xff0c;适用于Windows、macOS和Linux系统&#xff1a; 一、Windows系统下载与…

利用TinyML和IoT技术预测沙漠地区光伏电站清洁方法

论文标题 英文标题&#xff1a;Predictive method for cleaning photovoltaic plants in desert areas using TinyML and IoT technique 作者信息 A. Mellit, M. Chourouk&#xff1a;Faculty of Science and Technology, Renewable Energy laboratory, University of Jijel…

P3372 【模板】线段树 1

luoguP3372 【模板】线段树 1 题目描述 如题&#xff0c;已知一个数列&#xff0c;你需要进行下面两种操作&#xff1a; 将某区间每一个数加上 k k k。求出某区间每一个数的和。 输入格式 第一行包含两个整数 n , m n, m n,m&#xff0c;分别表示该数列数字的个数和操作…

Enigma Virtual Box封装客户端

1.输入可执行程序&#xff0c;另外命名输出可执行程序的输出程序。如图&#xff1a; 2.添加附带文件 这些文件包括可执行程序的库、文件、插件等。 如图&#xff1a;(这里包括文件或者文件夹) 3.点击process生成可执行文件 生成的执行文件可以放在桌面上单独运行。

Unity自动LOD工具AutoLOD Mesh Decimator的使用

最近在研究大批量物体生成&#xff0c;由于我们没有专业美术&#xff0c;在模型减面工作上没有人手&#xff0c;所以准备用插件来实现LOD功能&#xff0c;所以找到了AutoLOD Mesh Decimator这个插件。 1&#xff0c;导入插件后&#xff0c;我们拿个实验的僵尸狗来做实验。 空…

VMware彻底官宣免费!杀疯了!

话说最近这几个月&#xff0c;几家软件大佬这是怎么了&#xff0c;这怎么还开始卷免费了呢&#xff08;手动doge&#xff09;。 众所周知&#xff0c;就在上个月的时候&#xff0c;Jetbrains 刚官宣其旗下 WebStorm 和 Rider 两款软件开始对非商业用途全面免费&#xff0c;当时…

QML —— 拖拽测试 - 文本图片跑马灯Demo(附源码)

效果 说明 此代码可对文本及图片进行托转并放入被置方框内,在放置的文本框或图片框发生变化后,跑马灯也会在下一次运行时内容发生变化。 代码 main.qml import QtQuick 2.9 import QtQuick.Window 2.2 import QtQuick.Controls 2.0 import QtQuick.Layouts 1.14 import QtQu…

CDGA|企业数据治理:实务知识与理论思考的深度融合探索

在当今这个数据驱动的时代&#xff0c;企业数据已成为推动业务增长、优化决策制定和塑造竞争优势的关键因素。然而&#xff0c;随着数据量的爆炸性增长&#xff0c;如何有效管理和利用这些数据&#xff0c;确保数据的准确性、安全性与合规性&#xff0c;成为企业面临的一大挑战…

乐观锁和悲观锁的区别 使用 使用场景 | 图解

图解乐观锁和悲观锁的区别 & 实现 & 使用场景 文章目录 图解乐观锁和悲观锁的区别 & 实现 & 使用场景悲观锁synchronized 与 ReentrantLock 乐观锁CAS 机制版本号机制原子类 总结两种锁各自的使用场景 悲观锁 悲观主义者&#xff0c;认为这个资源不上锁&#x…

Linux初步引言(0)

文章目录 前言一、发展史UNIX发展史Linux发展史 二、开源精神三、Linux内核官网四、企业应用现状在服务器领域的发展在桌面领域的发展在移动嵌入式的发展Linux在云计算/大数据领域的发展 五、众多的发行版本DebianUbuntuCentOSKail Linux 六、何为操作系统&#xff1f;总结 前言…

Linux: C语言发起 DNS 查询报文

本文目录 使用 getaddrinfo()手动构造 DNS 查询报文DNS 查询部分&#xff08;Question Section&#xff09;QNAME (查询的域名)QTYPE (查询类型)QCLASS (查询类)Answer Section (答案部分) C语言代码发起 DNS 查询报文 使用 getaddrinfo() getaddrinfo() 是一个高层的接口&…

【Pytorch】神经网络介绍|激活函数|使用pytorch搭建方法

神经网络 神经网络介绍 概念 神经网络 人工神经网络ANN 也称神经网络NN 是一种模仿生物神经网络结构和功能的计算模型人脑可以看作是一个生物神经网络,由众多神经元连接而成,神经网络可以看作是模拟生物神经元的过程 输入层 input Layer: 输入x的那一层 输出层 output Laye…

【HarmonyOS NEXT】实战——登录页面

【HarmonyOS NEXT】实战——登录页面 在本文中&#xff0c;我们将深入探讨如何使用HarmonyOS NEXT来实现一个功能完备的登录页面。通过这个实战案例&#xff0c;你将结合页面布局、数据本地化存储、网络请求等多方面了解到HarmonyOS NEXT在构建现代应用时的强大能力和灵活性。…