C++20 协程:异步编程的新纪元

C++20 引入了协程(coroutines),这是一种全新的异步编程模型,使得编写异步代码变得更加简洁和直观。本文将详细介绍 C++20 协程的概念、功能演变及其在实际项目中的应用。通过本文,你将了解到协程的基本原理、语法和如何利用协程来简化异步编程。

1. 协程的概念

协程(coroutine) 是一种可以暂停执行的函数,它可以在执行过程中挂起(yield)并恢复执行。与传统的函数不同,协程可以在执行过程中多次进出,从而实现更高效的异步编程。

2. 功能演变
2.1 早期尝试
  • Boost.Coroutine:Boost 库早在 C++11 之前就提供了协程的支持。它通过底层的上下文切换来实现协程的功能。
  • P0333R11:C++ 标准委员会提出了一个关于协程的设计提案,最终在 C++20 中被采纳。
2.2 C++20 标准
  • co_await:用于等待一个协程表达式的结果。
  • co_yield:用于产生一个值,并暂时挂起协程。
  • co_return:用于从协程返回一个值。
3. C++20 协程的基本语法
3.1 协程函数

协程函数可以通过在返回类型前加上 ->std::suspend_neverstd::suspend_alwaysstd::suspend_never 来声明。

  • std::suspend_never:表示协程不会挂起。
  • std::suspend_always:表示协程总是挂起。
  • std::suspend_also:表示协程可能挂起也可能不挂起。
// 一个简单的协程函数
struct SimpleCoroutine {
    int operator()(int x) -> std::suspend_never {
        co_yield x * 2;
        co_return;
    }
};

int main() {
    SimpleCoroutine coro;
    auto result = coro(5); // 10
}
3.2 协程控制块

协程控制块(coroutine handle)用于管理协程的生命周期。

// 创建协程控制块
struct Coroutine {
    struct promise_type {
        int value;
        
        // 构造函数
        promise_type(int val) : value(val) {}

        // get_return_object
        auto get_return_object() {
            return *this;
        }

        // initial_suspend
        std::suspend_never initial_suspend() {}

        // final_suspend
        std::suspend_never final_suspend() noexcept {}

        // return
        void return_value(int val) {}

        // unhandled_exception
        void unhandled_exception() {}
    };

    coroutine_handle<promise_type> handle;

    Coroutine(coroutine_handle<promise_type> h) : handle(h) {}

    // 启动协程
    void start() {
        if (!handle.done()) {
            handle.resume();
        }
    }

    // 挂起协程
    void suspend() {
        if (!handle.done()) {
            handle.promise().value = 42;
            handle.suspend();
        }
    }

    // 等待协程
    int await() {
        if (!handle.done()) {
            handle.resume();
        }
        return handle.promise().value;
    }
};

// 使用协程
Coroutine::promise_type::value = 10;
Coroutine coro(Coroutine::promise_type::make_promise());
coro.start();
int result = coro.await(); // 42
4. 实战应用
4.1 异步 I/O

协程非常适合用于异步 I/O 操作。下面是一个简单的异步文件读取示例:

#include <iostream>
#include <fstream>
#include <coroutine>

struct AsyncFileReader {
    struct promise_type {
        std::string filename;
        std::ifstream file;
        std::string buffer;

        // get_return_object
        AsyncFileReader get_return_object() {
            return *this;
        }

        // initial_suspend
        std::suspend_never initial_suspend() {}

        // final_suspend
        std::suspend_never final_suspend() noexcept {}

        // return
        void return_value(std::string buf) {
            buffer = std::move(buf);
        }

        // unhandled_exception
        void unhandled_exception() {}

        void read_file() {
            file.open(filename);
            if (file.is_open()) {
                std::getline(file, buffer);
                co_return buffer;
            } else {
                co_return "";
            }
        }
    };

    coroutine_handle<promise_type> handle;

    AsyncFileReader(coroutine_handle<promise_type> h) : handle(h) {}

    std::string await_resume() {
        handle.resume();
        return handle.promise().buffer;
    }
};

AsyncFileReader async_read_file(std::string filename) {
    return AsyncFileReader{coroutine_handle<promise_type>::from_promise(promise_type{filename})};
}

int main() {
    auto result = async_read_file("example.txt").await_resume();
    std::cout << "Read data: " << result << std::endl;
}
4.2 异步网络编程

协程也非常适合用于异步网络编程。下面是一个简单的异步 HTTP 请求示例:

#include <iostream>
#include <coroutine>
#include <asio/awaitable.hpp>
#include <asio/io_context.hpp>
#include <asio/ip/tcp.hpp>
#include <asio/steady_timer.hpp>
#include <asio/co_spawn.hpp>

using namespace asio::awaitable;

struct HttpClient {
    asio::io_context& io;
    asio::ip::tcp::socket socket;

    HttpClient(asio::io_context& io) : io(io), socket(io) {}

    void connect(const std::string& host, const std::string& port) {
        asio::connect(socket, asio::ip::tcp::resolver(io).resolve(host, port));
    }

    std::string send_request(const std::string& request) {
        asio::write(socket, asio::buffer(request));
        asio::streambuf response_buffer;
        asio::read_until(socket, response_buffer, "\r\n\r\n");
        return asio::buffer_cast<const char*>(response_buffer.data());
    }
};

struct AsyncHttpClient {
    asio::io_context& io;
    HttpClient client;

    AsyncHttpClient(asio::io_context& io, const std::string& host, const std::string& port)
        : io(io), client(io) {
        client.connect(host, port);
    }

    std::string async_send_request(const std::string& request) {
        co_await asio::co_spawn(io, [&]() -> awaitable<void> {
            auto response = client.send_request(request);
            co_return;
        }, asio::use_awaitable);

        // 返回结果
        co_return "Response received";
    }
};

int main() {
    asio::io_context io;
    AsyncHttpClient client(io, "example.com", "80");

    auto result = client.async_send_request("GET / HTTP/1.1\r\nHost: example.com\r\n\r\n");
    std::cout << "Received response: " << result << std::endl;

    io.run();
}

结论

通过本文的介绍,你不仅了解了 C++20 协程的基本概念和语法,还学会了如何在实际项目中应用协程来简化异步编程。协程的引入极大地提升了 C++ 异步编程的能力,使得编写高性能、可维护的异步代码变得更加容易。希望这篇文章对你有所帮助,如果你有任何具体的问题或需要进一步的帮助,请随时留言交流!


通过本文的介绍,你将能够更好地理解和使用 C++20 协程,并在异步编程中有效地管理异步操作,提升应用程序的性能和用户体验。

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

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

相关文章

Linux 挂载磁盘与开机自动挂载操作指南

Linux 挂载磁盘与开机自动挂载操作指南 文章目录 Linux 挂载磁盘与开机自动挂载操作指南一 挂载磁盘1 查看硬盘信息2 新增数据盘执行分区3 新建分区4 创建一个主分区5 分区编号6 初始磁柱编号7 截止磁柱编号8 查看新建分区信息9 分区结果写入10 新分区同步操作系统11 设置新分区…

9.12 TFTP通信

客户端设计&#xff08;仅供参考&#xff09;&#xff1a; 下载本质&#xff1a;读取服务器发送的数据包&#xff0c;写入到本地文件 上传本质&#xff1a;读取本地文件内容&#xff0c;发送给服务器。 1、建立菜单选项&#xff0c;上传和下载。 2、上传功能函数&#xff1a; …

【程序分享】Warren Cowley Parameters 程序:表征短程有序的化学基序

分享一个 Warren Cowley Parameters 程序&#xff1a;表征短程有序的化学基序。 感谢论文的原作者&#xff01; 主要内容 “晶体材料的化学成分具有原子尺度的波动&#xff0c;可调节各种中尺度特性。建立此类材料的化学-微结构关系需要对这些化学波动进行适当的表征。然而&…

2024网安周今日开幕,亚信安全亮相30城

2024年国家网络安全宣传周今天在广州拉开帷幕。今年网安周继续以“网络安全为人民&#xff0c;网络安全靠人民”为主题。2024年国家网络安全宣传周涵盖了1场开幕式、1场高峰论坛、5个重要活动、15场分论坛/座谈会/闭门会、6个主题日活动和网络安全“六进”活动。亚信安全出席20…

yolov8-obb中存在的一个bug

yolov8支持OBB目标检测,且能提供较好的性能。 但是最近在使用yolov8-obb的过程中,发现yolov8-obb存在一个bug。即训练数据如果包含不带旋转角度的水平目标时,训练出的模型,经常会输出垂直的检测框,需要旋转90度以后才能得到最终结果。把yolov8-obb相关的源码阅读一遍才发…

【数学建模】2024数学建模国赛经验分享

文章目录 一、关于我二、我的数模历程三、经验总结&#xff1a; 一、关于我 我的CSDN主页&#xff1a;https://gxdxyl.blog.csdn.net/ 2020年7月&#xff08;大二结束的暑假&#xff09;开始在CSDN写作&#xff1a; 阿里云博客专家&#xff1a; 接触的领域挺多的&#xff…

HTML 转 PDF API 接口

HTML 转 PDF API 接口 网络工具 / 文件处理 支持网页转 PDF 高效生成 PDF / 提供永久链接。 1. 产品功能 超高性能转换效率&#xff1b;支持将传递的 HTML 转换为 PDF&#xff0c;支持转换 HTML 中的 CSS 格式&#xff1b;支持传递网站 URL&#xff0c;直接转换页面成对应的 …

Java实现生成验证码实战

文章目录 需求描述思想思路实现代码实现效果 在实际项目中&#xff0c;管理端的登录&#xff0c;会涉及验证码的校验&#xff0c;简单的数字与字母组合形式&#xff0c;在Java中要如何生成与实现&#xff0c;记录下来&#xff0c;方便备查。 需求描述 生成8位的由数字、大写字…

【零基础学习CAPL】——CRC值监控测试

🙋‍♂️【零基础学习CAPL】系列💁‍♂️点击跳转 ——————————————————————————————————–—— 从0开始学习CANoe使用 从0开始学习车载车身 相信时间的力量 星光不负赶路者,时光不负有心人。 目录 1.概述2.需求介绍3.算法4.逻辑判断5.测…

ARCGIS PRO DSK MapTool

MapTool用于自定义地图操作工具&#xff0c;使用户能够在ArcGIS Pro中执行特定的地图交互操作。添加 打开MapTool1.vb文件&#xff0c;可以看到系统已经放出MapTool1类&#xff1a; Public Sub New()将 IsSketchTool 设置为 true 以使此属性生效IsSketchTool TrueSketchTyp…

为了准确计算延迟退休时间,我做了一个退休年龄计算器

延迟退休计算方法 原本退休分为三种情况&#xff0c;男性&#xff0c;女工人&#xff0c;女干部 男性&#xff1a;退休年龄为60岁。女干部&#xff1a;退休年龄为55岁。女工人&#xff1a;退休年龄为50岁。 现在延迟以后&#xff08;根据2024年9月13日公布的规则&#xff09…

一次开发,多端部署--实例二

一、视觉风格 1、分层参数 使用了分层参数后&#xff0c;当系统切换深色模式时&#xff0c;字体和背景也可以自适应。 Row() {Column() {Text(分层参数)// 分层参数在sysResource包&#xff0c;属于系统参数&#xff0c;全局可用.fontColor($r(sys_color.ohos_id_color_text_pr…

JavaScript模块化——ES6模块化规范

作者&#xff1a;CSDN-PleaSure乐事 欢迎大家阅读我的博客 希望大家喜欢 使用环境&#xff1a;vscode Chrome浏览器 1.ES6 1.1ES6介绍 ES6的全称是ECMAScript 6&#xff0c;也称为ES2015&#xff0c;是JavaScript的一个重要版本&#xff0c;它引入了许多新特性和改进&#xf…

云计算实训43——部署k8s基础环境、配置内核模块、基本组件安装

一、前期系统环境准备 1、关闭防火墙与selinux [rootk8s-master ~]# systemctl stop firewalld[rootk8s-master ~]# systemctl disable firewalldRemoved symlink /etc/systemd/system/multi-user.target.wants/firewalld.service. Removed symlink /etc/systemd/system/dbus…

云渲染与AI渲染分别是什么?两者的优势对比

云渲染和AI渲染是两种先进的渲染技术&#xff0c;它们各自具有独特的优势和应用场景。下面针对两种情况来简单说明下。 1、云渲染&#xff1a; - 定义&#xff1a;云渲染是一种利用远程服务器(云端)来处理和生成渲染效果的技术。它允许用户将计算密集型的任务转移到云端&#…

uniapp网络延迟优化之骨架屏

文章目录 前言uniapp网络延迟优化之骨架屏 一、骨架屏是什么&#xff1f;二、使用步骤1.在微信开发者工具生成骨架屏文件2.转成vue组件3.组件中使用4.效果展示4.开发时遇到的问题&#xff1f; 总结 前言 uniapp网络延迟优化之骨架屏 一、骨架屏是什么&#xff1f; 骨架屏的主…

C++ | Leetcode C++题解之第403题青蛙过河

题目&#xff1a; 题解&#xff1a; class Solution { public:bool canCross(vector<int>& stones) {int n stones.size();vector<vector<int>> dp(n, vector<int>(n));dp[0][0] true;for (int i 1; i < n; i) {if (stones[i] - stones[i -…

卷积神经网络经典模型架构简介

【图书推荐】《PyTorch深度学习与企业级项目实战》-CSDN博客 《PyTorch深度学习与企业级项目实战&#xff08;人工智能技术丛书&#xff09;》(宋立桓&#xff0c;宋立林)【摘要 书评 试读】- 京东图书 (jd.com) ImageNet是一个包含超过1 500万幅手工标记的高分辨率图像的数据…

Redis——常用数据类型List

目录 List列表常用命令lpushlpushxrpushrpushlrangelpoprpoplindexlinsertllenlremltrim key start stoplset 阻塞版本命令blpopbrpop list的编码方式list的应用 List列表 Redis中的list相当于数组&#xff0c;或者 顺序表&#xff0c;一些常用的操作可以通过下面这张图来理解…

ubuntu 遇到的一些问题及解决办法

一、“E: 无法定位软件包” 产生该问题的安全提示原因有很多。如网络链接问题、apt 源过期了。 解决方法&#xff1a; 1. 备份 /etc/apt/sources.list 文件 cp /etc/apt/sources.list /etc/apt/sources.list.old 2. 打开文件 gedit /etc/apt/sources.list 3. 添加清华源…