vs2019 16.8更新之后的 C++20 协程co_yield用法
由于搜索出来的帖子,都是老版本的实验协程,很多老的代码已经失去参考性,并且很复杂,所以就自己研究了一下。
1 #include <iostream>
2 #include <coroutine>
3 #include <thread>
4
5 template<typename _Ty>
6 struct cocontext {
7 struct promise_type;
8 using _Hty = std::coroutine_handle<promise_type>;
9 struct promise_type {
10 // 只要一个函数的返回值是 cocontext<T>,这个函数内存在co_await co_yield co_return这3个语法糖
11 // 就会第一时间调用 get_return_object
12 cocontext get_return_object() {
13 return { _Hty::from_promise(*this) };
14 }
15
16 // 协程异常时抛出到这里,关于异常,见仁见智,反正我是不用的。
17 void unhandled_exception() { std::terminate(); }
18
19 // co_await co_yield co_return 之前回调,只回调一次,它在get_return_object之后回调
20 auto initial_suspend() { return std::suspend_never{}; }
21
22 // 协程函数返回之后,回调这里
23 auto final_suspend() { return std::suspend_always{}; }
24
25 // 如果函数里没有co_return,必须实现return_void,与之相对的还有一个return_value,类似yield_value
26 void return_void() {}
27
28 // 如果函数里有co_yield,必须实现yield_value,这其实并不难理解,co_yield把数据传到参数,然后我储存在_Val里而已
29 auto yield_value(const _Ty &val) {
30 _Val = val;
31 return std::suspend_always{};
32 }
33 _Ty _Val;
34 };
35
36 // 如果 await_ready 返回true,它就会继续执行,所以理论上来说,要模拟异步场景,都只会是return false
37 bool await_ready() const{ return false; }
38
39 // await_suspend是可以有参数的,它还可以是 void await_suspend(std::coroutine_handle<cocontext> handle);
40 // 在co_await co_yield co_return时,首先会调用这里,也就是可以根据情况直接在这里进行handle.resume();
41 void await_suspend() {}
42
43 // 下面是我自己的实现
44 cocontext& resume() { if (!_Handle.done())_Handle.resume(); return *this; }
45 operator _Ty() const { return _Handle.promise()._Val; }
46
47 _Hty _Handle;
48 };
49
50 int main()
51 {
52 auto test = []()->cocontext<int> { for (int i = 0; i < 10; i++) co_yield i; };
53 auto c = test();
54 std::cout << (int)c << std::endl; // 0
55 std::cout << (int)c.resume() << std::endl; // 1
56 std::cout << (int)c.resume() << std::endl; // 2
57 std::cout << (int)c.resume() << std::endl; // 3
58 std::cout << (int)c.resume() << std::endl; // 4
59
60 std::thread([&]() {
61 // 协程真正有意思的地方是,它可以由不同的线程去resume,这会很有意义。
62 std::cout << (int)c.resume() << std::endl; // 5
63 std::cout << (int)c.resume() << std::endl; // 6
64 std::cout << (int)c.resume() << std::endl; // 7
65 std::cout << (int)c.resume() << std::endl; // 8
66 std::cout << (int)c.resume() << std::endl; // 9
67
68 // 这里依旧输出9,上面一个c.resume()之后,test函数已经跳出循环返回了,已经满足了_Handle.done(),不会再继续真正的_Handle.resume();
69 std::cout << (int)c.resume() << std::endl;
70 }).join();
71
72 }
73
74 /*
75 后话:
76 C++ 20所谓的协程,实际上在我看来,更像是语法糖内部包含了一堆回调函数
77 并且这些回调函数得让程序员自己去完全实现,这确实是C++的风格,
78 但说实在的,我相信搞得清楚这些东西的人,都不会太喜欢这种风格。
79
80 虽然我的例子中 cocontext是基本可重用的类型,但我依旧没感觉这种形式为我的程序带来了多少便利。
81 情况还是那个情况,这个东西用起来并不方便,程序员从一开始就要考虑到所有细节,
82 可能最终很多细节考虑之后,用起来还是一大堆代码,到时候可能写着写着把递归逻辑搞成线性逻辑了,然后感叹一句,“我是SB,我TM用个P的协程!”。
83 */
vs2019 16.8更新之后的 C++20 协程co_yield用法的更多相关文章
- C++20协程实例:携程化的IOCP服务端/客户端
VC支持协程已经有一段时间了,之前一直想不明白协程的意义在哪里,前几天拉屎的时候突然灵光一闪: 以下是伪代码: task server() { for (;;) { sock_context s = ...
- ubuntu 16.04 更新 gcc/g++ 4.9.2
ubuntu 转载 2016年10月12日 :: 标签:ubuntu /g++ /gcc [html] view plain copy sudo dpkg -l g++ 最近在学C++primer , ...
- python基础(16)-进程&线程&协程
进程之multiprocessing模块 Process(进程) Process模块是一个创建进程的模块,借助这个模块,就可以完成进程的创建. 介绍 初始化参数 Process([group [, t ...
- Python自动化运维之16、线程、进程、协程、queue队列
一.线程 1.什么是线程 线程是操作系统能够进行运算调度的最小单位.它被包含在进程之中,是进程中的实际运作单位. 一条线程指的是进程中一个单一顺序的控制流,一个进程中可以并发多个线程,每条线程并行执行 ...
- golang学习笔记20 一道考察对并发多协程操作一个共享变量的面试题
golang学习笔记20 一道考察对并发多协程操作一个共享变量的面试题 下面这个程序运行的能num结果是什么? package main import ( "fmt" " ...
- 【流畅的python】16.1 - 生成器如何进化成协程
在生成器中加入yield关键字后,生成器调用方可以向生成器传入值,只需要使用.send(...)方法就可以传送数据.发送的数据会成为生成器函数中yield表达式的值.所以生成器可以作为协程使用. 协程 ...
- python day 20: 线程池与协程,多进程TCP服务器
目录 python day 20: 线程池与协程 2. 线程 3. 进程 4. 协程:gevent模块,又叫微线程 5. 扩展 6. 自定义线程池 7. 实现多进程TCP服务器 8. 实现多线程TCP ...
- 比物理线程都好用的C++20的协程,你会用吗?
摘要:事件驱动(event driven)是一种常见的代码模型,其通常会有一个主循环(mainloop)不断的从队列中接收事件,然后分发给相应的函数/模块处理.常见使用事件驱动模型的软件包括图形用户界 ...
- linux_ubuntu 16.04 更新wifi驱动_无法链接wifi问题
ubuntu kylin ubuntu kylin ubuntu kylin wifi 这个很好解决的,16.04 默认 没有使用wifi驱动设备,默认选择的是:不使用设备1.进入到,软件和更新 -- ...
随机推荐
- 大牛针对零基础入门c语言详解指针(超详细)
C语言指针说难不难但是说容易又是最容易出错的地方,因此不管是你要做什么只要用到C指针你就跳不过,今天咱们就以 十九个例子来给大家简单的分析一下指针的应用,最后会有C语言视频资料提供给大家更加深入的参考 ...
- Apache Kafka 学习笔记
1. 介绍Kafka是由Apache软件基金会开发的一个开源流处理平台,由Scala和Java编写.Kafka是一种高吞吐量的分布式发布订阅消息系统,它可以处理消费者在网站中的所有动作流数据. 这种动 ...
- hdu 5179 beautiful number(构造,,,,)
题意: 一个如果称作是漂亮数,当且仅当满足: 每一位上的数字是[1,9],从高到时低数字大小降序,且有di%dj=0(i<j) 例:931 给一个区间[L,R],问这个区间里有多少个漂亮数. 1 ...
- docker 使用报错的相关问题
docker 创建本地主机实例Virtualbox 驱动报错,显示没有下载这个驱动 解决方案,下载virtuabox. https://www.cnblogs.com/effortday/p/1502 ...
- upload-labs通关攻略(1-11关)
upload-labs通关攻略 upload-labs是练习文件上传很好的一个靶场,建议把upload-labs关卡全部练习一遍 1.下载安装 下载地址 链接:https://pan.baidu.co ...
- Kioskcached(1)之 Memcached & Redis & Kioskcached 性能测试对比
前言:本文仅仅是作者自己在学习过程中的一次实验而已,或许因为各种因素会导致实验结果与你之前的认知不太一样,因此请你带着批判的眼光看待本文(本文不具有实际环境的参考性). 一:测试目的 在了解了一些No ...
- SpringBoot2.x异步任务EnableAsync
1.springboot启动类里面使用@EnableAsync注解开启异步功能 @EnableAsync public class Demo001Application { public static ...
- CentOS8安装VNC-Server,并使用VNC Viewer连接
1.查看系统信息 # 查看red-hat版本信息 cat /etc/redhat-release CentOS Linux release 8.0.1905 (Core) 2.安装VNC Server ...
- Linux USB (目录)
1.USB 总线简介 2.USB 协议分析 3.USB Host 详解 4.USB Device 详解 5.usbip (USB Over IP) 使用实例
- C/C++ QT QChart 绘制组件应用
QtCharts 组件是QT中提供图表绘制的模块,该模块可以方便的绘制常规图形,Qtcharts 组件基于GraphicsView模式实现,其核心是QChartView和QChart的二次封装版. 在 ...