C++ 的协程

现代C++中的协程(coroutines)是C++20引入的一项重大语言特性,它们允许函数在执行过程中可以暂停并稍后从暂停点恢复执行。协程提供了一种控制流机制,使得函数可以包含多个入口点和出口点,这与传统的单入口、单出口的函数模型形成了鲜明对比。

协程的核心思想是“可恢复的函数”,即在执行过程中,函数可以“挂起”其执行状态,并在将来的某个时间点“恢复”执行。这种机制使得编写异步代码变得更加直观和易于管理,因为你可以使用类似于同步代码的结构来表达异步操作。

在C++中,协程通过特定的语法和一组标准库组件来实现。以下是一些关键点:

  1. 协程句柄(coroutine handle)
    协程句柄是一个轻量级的对象,它封装了与协程执行相关的状态和控制信息。你可以使用协程句柄来暂停、恢复或销毁协程。

  2. 协程承诺类型(promise type)
    每个协程都有一个与之关联的承诺类型(通常是一个模板类),它定义了协程的返回类型和状态。承诺类型负责在协程挂起和恢复时管理其状态,并处理与协程相关的异常和返回值。

  3. co_awaitco_yieldco_return
    这三个关键字是C++协程语法的核心。co_await用于挂起协程并等待一个awaitable对象的完成;co_yield用于在生成器中逐个产生值;co_return用于从协程返回结果。

  4. 标准库支持
    C++20标准库提供了一些与协程相关的组件,如std::suspend_alwaysstd::suspend_neverstd::coroutine_handle等,以及用于定义协程行为的模板类(如std::generatorstd::async_generator,尽管这些可能在未来的标准中有所变化或扩展)。

  5. 异步编程
    协程最显著的应用之一是异步编程。通过协程,你可以编写看起来像是同步调用的异步代码,从而简化异步操作的复杂性。

  6. 生成器
    协程还可以用于实现生成器,即按需产生一系列值的函数。这与迭代器类似,但生成器允许更复杂的逻辑和状态管理。

  7. 实现细节
    协程的实现涉及编译器对特定语法结构的支持,以及标准库提供的底层机制。编译器需要识别协程函数,并在编译时生成适当的代码来处理协程的挂起和恢复。

使用协程涉及几个关键步骤,包括定义协程函数、使用特定的协程语法(如co_awaitco_yieldco_return),以及可能定义自定义的promise类型来管理协程的状态和返回值。以下是一个基本的指南,帮助你开始在现代C++中使用协程。

1. 编译器支持

首先,确保你的编译器支持C++20协程。GCC 10及以上版本、Clang 11及以上版本以及MSVC(Visual Studio 2019版本16.6及以上)都提供了对C++20协程的支持。

2. 包含必要的头文件

在你的C++源文件中包含必要的头文件。对于基本的协程功能,你可能需要包含<coroutine>头文件。

#include <coroutine>

3. 定义协程函数

协程函数使用co_awaitco_yield(对于生成器)或co_return来控制其执行流程。下面是一个简单的例子,展示了如何使用co_await来挂起和恢复协程的执行。

#include <coroutine>
#include <iostream>
#include <thread>
#include <chrono>// 一个简单的awaitable对象,它会在一段时间后“完成”
struct simple_awaitable {bool await_ready() const noexcept { return false; } // 协程需要挂起void await_suspend(std::coroutine_handle<>) const noexcept {} // 挂起时的操作(这里为空)void await_resume() const noexcept {} // 恢复时的操作(这里为空)// 模拟异步等待(例如,I/O操作)static simple_awaitable await_now() { return {}; }
};// 协程函数示例
struct MyCoroutine {struct promise_type {MyCoroutine get_return_object() { return MyCoroutine{std::coroutine_handle<promise_type>::from_promise(*this)}; }std::suspend_always initial_suspend() { return {}; }std::suspend_always final_suspend() noexcept { return {}; }void return_void() {}void unhandled_exception() { std::exit(1); } // 处理异常(这里简单退出程序)};std::coroutine_handle<promise_type> handle;MyCoroutine(std::coroutine_handle<promise_type> h) : handle(h) {}~MyCoroutine() { if (handle) handle.destroy(); }void resume() { handle.resume(); }
};MyCoroutine example_coroutine() {std::cout << "Before await\n";co_await simple_awaitable::await_now(); // 挂起协程std::cout << "After await\n";
}int main() {auto coro = example_coroutine();std::cout << "Main thread continues\n";std::this_thread::sleep_for(std::chrono::seconds(1)); // 模拟主线程做一些其他工作coro.resume(); // 恢复协程的执行return 0;
}

4. 使用协程

在上面的例子中,example_coroutine是一个协程函数,它使用co_await来挂起其执行。在main函数中,我们创建了协程对象coro,并在稍后通过调用coro.resume()来恢复其执行。

5. 自定义promise类型(可选)

在上面的例子中,我们定义了MyCoroutine结构和一个简单的promise_type来管理协程的状态。对于更复杂的协程,你可能需要定义自己的promise类型来处理返回值、异常和协程的生命周期。

6. 注意事项

  • 协程是编译器和语言特性的结合,因此确保你的编译器和标准库支持C++20协程。
  • 协程的状态管理是通过promise类型来实现的,因此理解promise类型的工作原理对于编写有效的协程至关重要。
  • 协程的异步性质意味着它们可以与其他线程交互,因此需要注意线程安全和同步问题。
  • 尽管C++20引入了协程的基本框架和语法,但协程的完整功能和最佳实践仍在不断发展和完善中。因此,在使用协程时,建议查阅最新的C++标准文档和编译器文档,以获取最新的信息和最佳实践。

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

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

相关文章

Leecode热题100-35.搜索插入位置

给定一个排序数组和一个目标值&#xff0c;在数组中找到目标值&#xff0c;并返回其索引。如果目标值不存在于数组中&#xff0c;返回它将会被按顺序插入的位置。 请必须使用时间复杂度为 O(log n) 的算法。 示例 1: 输入: nums [1,3,5,6], target 5 输出: 2示例 2: 输入:…

Axure设计之文本编辑器制作教程

文本编辑器是一个功能强大的工具&#xff0c;允许用户在图形界面中创建和编辑文本的格式和布局&#xff0c;如字体样式、大小、颜色、对齐方式等&#xff0c;在Web端实际项目中&#xff0c;文本编辑器的使用非常频繁。以下是在Axure中模拟web端富文本编辑器&#xff0c;来制作文…

【MySQL 保姆级教学】事务的隔离级别(详细)--下(13)

事务的隔离级别 1. 如何理解事务的隔离性2. 事务隔离级别的分类3. 查看和设置事务隔离级别3.1 全局和会话隔离级别3.2 查看和设置隔离级别 4. 事务隔离级别的演示4.1 读未提交&#xff08;Read Uncommitted&#xff09;4.2 读已提交&#xff08;Read Committed&#xff09;4.3 …

大厂的 404 页面都长啥样?看看你都见过吗~~~

当我们浏览网页时&#xff0c;不小心走错路径或打开一个已被移除的页面时&#xff0c;常会遇到“404页面”。这时&#xff0c;普通网站往往只会显示冷冰冰的“404 Not Found”&#xff0c;但大厂们却能把404页面玩出花来。国内互联网大厂的404页面不仅独特&#xff0c;而且设计…

acwing算法基础02一高精度,前缀和,差分

#include <iostream> #include <vector> using namespace std;const int N 1e6 10; //模板 CABvector<int> add(vector<int> &A,vector <int> &B) {vector<int> C;int t 0; // 用来保存每位的和&#xff08;包括进位&#xff…

WebAssembly在现代Web开发中的应用

&#x1f493; 博客主页&#xff1a;瑕疵的CSDN主页 &#x1f4dd; Gitee主页&#xff1a;瑕疵的gitee主页 ⏩ 文章专栏&#xff1a;《热点资讯》 WebAssembly在现代Web开发中的应用 WebAssembly在现代Web开发中的应用 WebAssembly在现代Web开发中的应用 引言 WebAssembly 概述…

06.VSCODE:备战大项目,CMake专项配置

娇小灵活的简捷配置不过是年轻人谈情说爱的玩具&#xff0c;帝国大厦的构建&#xff0c;终归要交给CMake去母仪天下。一个没有使用 CMake 的 C 项目&#xff0c;就像未来世界里的一台相声表演&#xff0c;有了德纲却无谦&#xff0c;观众笑着遗憾。—— 语出《双城记》作者&…

从社交媒体到元宇宙:Facebook未来发展新方向

Facebook&#xff0c;作为全球最大的社交媒体平台之一&#xff0c;已经从最初的简单互动工具发展成为一个跨越多个领域的科技巨头。无论是连接人与人之间的社交纽带&#xff0c;还是利用大数据、人工智能等技术为用户提供个性化的体验&#xff0c;Facebook一直引领着社交网络的…

【go从零单排】JSON序列化和反序列化

&#x1f308;Don’t worry , just coding! 内耗与overthinking只会削弱你的精力&#xff0c;虚度你的光阴&#xff0c;每天迈出一小步&#xff0c;回头时发现已经走了很远。 &#x1f4d7;概念 在 Go 语言中&#xff0c;处理 JSON 数据主要依赖于 encoding/json 包。这个包提…

vue2.7.14 + vant + vue cli脚手架转vite启动运行问题记录

文章目录 前言方案一&#xff08;借用插件转换&#xff09;启动命令&#xff0c;转换方案一转换遇到的问题 方案二&#xff08;手动调整&#xff09;方案两者对比小结 前言 vue cli 脚手架转成vite启动 简单说说这个项目的一些底层基本结构哈&#xff0c;以及写这篇博客的目的…

Linux 常用操作指令大揭秘(下)

&#x1f31f;快来参与讨论&#x1f4ac;&#xff0c;点赞&#x1f44d;、收藏⭐、分享&#x1f4e4;&#xff0c;共创活力社区。 &#x1f31f; &#x1f6a9;用通俗易懂且不失专业性的文字&#xff0c;讲解计算机领域那些看似枯燥的知识点&#x1f6a9; 目录 &#x1f4af;…

HBase使用create创建表时报错ERROR: KeeperErrorCode = NoNode for /hbase/master

场景模拟 1. 正常情况 模拟ERROR: KeeperErrorCode NoNode for /hbase/master错误场景。 正常情况下创建hbase表如下图所示。 2. 删除hbase集群的zk节点 进入zookeeper客户端。 zkCli.sh删除hbase的zk节点。 deleteall /hbase退出zookeeper客户端。 quit3. 重启hbase集…

【设计模式】行为型模式(二):策略模式、命令模式

行为型模式&#xff08;二&#xff09;&#xff1a;策略模式、命令模式 3.策略模式&#xff08;Strategy&#xff09;3.1 示例3.1.1 定义策略接口3.1.2 实现具体策略3.1.3 定义上下文类3.1.4 客户端代码3.1.5 输出结果 3.2 总结3.2.1 优点3.2.2 缺点 4.命令模式&#xff08;Com…

java八股-jvm入门-程序计数器,堆,元空间,虚拟机栈,本地方法栈,类加载器,双亲委派,类加载执行过程

文章目录 PC Register堆虚拟机栈方法区(Metaspace元空间双亲委派机制类加载器 类装载的执行过程 PC Register 程序计数器&#xff08;Program Counter Register&#xff09;是 Java 虚拟机&#xff08;JVM&#xff09;中的一个组件&#xff0c;它在 JVM 的内存模型中扮演着非常…

Python →爬虫实践

爬取研究中心的书目 现在&#xff0c;想要把如下网站中的书目信息爬取出来。 案例一 耶鲁 Publications | Yale Law School 分析网页&#xff0c;如下图所示&#xff0c;需要爬取的页面&#xff0c;标签信息是“<p>”&#xff0c;所以用 itemssoup.find_all("p&…

【Linux】-学习笔记03

第十一章-管理Linux软件包和进程 1.源码下载安装软件 1.1概念 源码文件&#xff1a;程序编写者使用C或C等语言编写的原始代码文本文件 源码文件使用.tar.gz或.tar.bz2打包成压缩文件 1.2特点 源码包可移植性好&#xff0c;与待安装软件的工作环境依赖性不大 由于有编译过程…

排序算法 - 冒泡

文章目录 1. 冒泡排序1.1 简介1.2 基本步骤&#xff1a;1.3 示例代码&#xff08;C&#xff09;1.4 复杂度分析1.5 动画展示 1. 冒泡排序 1.1 简介 冒泡排序&#xff08;Bubble Sort&#xff09;是一种简单的排序算法&#xff0c;其基本思想是通过相邻元素的比较和交换&#…

前端请求后端php接口跨域 cors问题

只需要后端在网站的入口文件 一般都是 index.php 加上 这几行代码就可以了 具体的参数可以根据需要去修改 header("Access-Control-Allow-Origin: *"); header(Access-Control-Allow-Methods: GET, POST, PUT, DELETE, OPTIONS); header(Access-Control-Allow-Heade…

Django基础用法+Demo演示

Django快速上手 参考: Django快速上手 再写几个页面 编辑demo1/urls.py, 添加URL和视图函数映射 urlpatterns [path(index/, views.index),path(user/list/, views.user_list),path(user/add/, views.user_add), ]编辑app01/views.py&#xff0c;添加几个函数 from djang…

数据集标注txt文件读取小工具

最近在看遥感图像目标检测相关的yolo10&#xff0c;自己在网上下载了数据集跑模型&#xff0c;但是跑出来的结果与数据集出处的论文介绍分类有些不同&#xff0c;只出现了分类0的情况&#xff0c;怀疑是标注有问题&#xff0c;但是数据集太大&#xff0c;于是做了个小工具对标注…