由于搜索出来的帖子,都是老版本的实验协程,很多老的代码已经失去参考性,并且很复杂,所以就自己研究了一下。

 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用法的更多相关文章

  1. C++20协程实例:携程化的IOCP服务端/客户端

    VC支持协程已经有一段时间了,之前一直想不明白协程的意义在哪里,前几天拉屎的时候突然灵光一闪: 以下是伪代码: task server() { for (;;) { sock_context s = ...

  2. 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 , ...

  3. python基础(16)-进程&线程&协程

    进程之multiprocessing模块 Process(进程) Process模块是一个创建进程的模块,借助这个模块,就可以完成进程的创建. 介绍 初始化参数 Process([group [, t ...

  4. Python自动化运维之16、线程、进程、协程、queue队列

    一.线程 1.什么是线程 线程是操作系统能够进行运算调度的最小单位.它被包含在进程之中,是进程中的实际运作单位. 一条线程指的是进程中一个单一顺序的控制流,一个进程中可以并发多个线程,每条线程并行执行 ...

  5. golang学习笔记20 一道考察对并发多协程操作一个共享变量的面试题

    golang学习笔记20 一道考察对并发多协程操作一个共享变量的面试题 下面这个程序运行的能num结果是什么? package main import ( "fmt" " ...

  6. 【流畅的python】16.1 - 生成器如何进化成协程

    在生成器中加入yield关键字后,生成器调用方可以向生成器传入值,只需要使用.send(...)方法就可以传送数据.发送的数据会成为生成器函数中yield表达式的值.所以生成器可以作为协程使用. 协程 ...

  7. python day 20: 线程池与协程,多进程TCP服务器

    目录 python day 20: 线程池与协程 2. 线程 3. 进程 4. 协程:gevent模块,又叫微线程 5. 扩展 6. 自定义线程池 7. 实现多进程TCP服务器 8. 实现多线程TCP ...

  8. 比物理线程都好用的C++20的协程,你会用吗?

    摘要:事件驱动(event driven)是一种常见的代码模型,其通常会有一个主循环(mainloop)不断的从队列中接收事件,然后分发给相应的函数/模块处理.常见使用事件驱动模型的软件包括图形用户界 ...

  9. linux_ubuntu 16.04 更新wifi驱动_无法链接wifi问题

    ubuntu kylin ubuntu kylin ubuntu kylin wifi 这个很好解决的,16.04 默认 没有使用wifi驱动设备,默认选择的是:不使用设备1.进入到,软件和更新 -- ...

随机推荐

  1. Oracle 11g 新建用户

    create user XXXuser identified by XXXpassword;--创建用户XXXuser,设置初始密码XXXpassword alter user XXXuser ide ...

  2. ES查询区分大小写

    ES查询区分大小写 ES查询在默认的情况下是不区分大小写的,在5.0版本之后将string类型拆分成两种新的数据类型,text用于全文搜索(模糊搜索),keyword用于关键字搜索(精确搜索). 注意 ...

  3. hdu 5166 Missing number(。。。)

    题意: 有一个排列,但少了两个数.给你少了这两个数的排列.找出是哪两个数. 思路: 看代码,,, 代码: int a[1005]; int main(){ int T; cin>>T; w ...

  4. java中Map及Map.Entry详解

    Map是java中的接口,Map.Entry是Map的一个内部接口. Map提供了一些常用方法,如keySet().entrySet()等方法. keySet()方法返回值是Map中key值的集合:e ...

  5. Trap (陷入/中断) 源码解析

    用户空间和内核空间之间的切换通常称为trap trap的三种形式 系统调用引发 异常发生 设备中断 (时间中断.IO中断.网络中断等) supervise mode的权限 用户态和内核态之间的到底有什 ...

  6. Linux 兴趣小组2016免试题 第四关揭秘

    Linux 兴趣小组2016免试题 点这里 首先贴出第四关链接Linux 兴趣小组2016免试题 第四关 第四关: 进入网址我们看到的是4张扑克牌K,这是什么意思? 要我斗地主?好了,还是乖乖的先查看 ...

  7. 元素显示模式:块元素 & 行内元素 & 行内块元素

    元素显示模式 前言 了解元素的显示模式可以更好的让我们布局页面.了解显示模式需要学习以下三个方面 什么是元素的显示模式 元素显示模式的分类 元素显示模式的转换 什么是元素显示模式 元素显示模式就是元素 ...

  8. mbatis动态sql中传入list并使用

    <!--Map:不单单forech中的collection属性是map.key,其它所有属性都是map.key,比如下面的departmentId --> <select id=&q ...

  9. stom消费kafka消息速度慢的问题

    原来代码如下 KafkaSpoutConfig<String, String> kafkaSpoutConfig = KafkaSpoutConfig.builder(kafka_serv ...

  10. 1. 处理静态资源 2. controller如何接受请求得参数 3. 如何把controller得数据保存到view. 4. 在controller如何完成重定向到指定路径 5. controller返回json数据

    1. 1. 处理静态资源2. controller如何接受请求得参数3. 如何把controller得数据保存到view.4. 在controller如何完成重定向到指定路径5. controller ...