c++协待

co_await expr

co_await(协待)C++20新关键字.上面表示等待懒求值任务.不希望该任务阻塞调用者,用协待挂起任务.调用者继续.任务完成后,协待返回协程的结果.有两个作用:挂起协程及取协程返回值.
协待等待任务时,不阻塞调用者,就是协程异步同步的关键.
协待类似调用函数.对象有()时是可调用的.对象支持协待()时,就是可等待的.协待后的可等待式.

总是挂起
协待 从不挂起;
空 测试(){
  协待 总是挂起{};
}//错误,缺少承诺类型.

协待必须在协程中,协程必须要有承诺类型.通过它返回内外通信对象.

构 任务{//任务
  构 承诺类型{
    任务 取中对象(){中{协程句柄<任务::承诺类型>::从承诺(*本)};}
    从不挂起 初始挂起(){中{};}
    从不挂起 止挂起()无异{中{};}
    空 中空(){}
    空 未处理异常(){}
  };

  协程句柄<任务::承诺类型>句柄_;
};

任务 测试(){
  协待 总是挂起{};//.1
}

.1转换为前面见过的模板框架代码:

{//
  动&&值=总是挂起{};
//先取表达式值.
  动&&w=取可等待(承诺,静转<推导(值)>(值));
//然后取`可等待`.
  动&&等待器=取等待器(静转<推导(w)>(w));
  如(!等待器.准备好协())
  {
    用 句柄型=实验::协程句柄<P>;
    用 等待挂起果型=推导(等待器.挂起协(句柄型::从承诺(p)));
    <挂起协程>
    如 常式(是空值<等待挂起果型>)
    {
      等待器.挂起协(句柄型::从承诺(p));
      <返回到调用者或恢复者>
    } 异 {
      静断(是相同值<等待挂起果型,极>,
         "挂起协()必须中'空'或'极'.");

      如(等待器.挂起协(句柄型::从承诺(p)))
      {
        <返回到调用者或恢复者>
      }
    }

    <恢复点>
  }

  中 等待器.恢复协();
}
//取可等待的代码
元<类 P,类 T>
推导(动)取可等待(P&承诺,T&&式)
{
  如 常式(有任何等待转换成员值<P>)
    中 承诺.等待转换(静转<T&&>(式));
  异
    中 静转<T&&>(式);
}

如果协程的承诺中定义了await_transform,那么就调用来得到可等待,否则就按可等待返回自己,这里,未在task里面定义await_transform,因此总是挂起将为可等待,实现代码:

构 总是挂起{
    常式 极 准备好协()常 无异 中 假;
    常式 空 挂起协(协程句柄<>)常 无异{}
    常式 空 恢复协()常 无异{}
};

取了总是挂起后,再根据它来取等待器,代码:

元<类 W>
推导(动)取等待器(W&&w)
{
  如 常式(有成员操作符协待值<W>)
    中 静转<W&&>(w).符号 协待();
    //有自身的`协待
  异 如 常式(无成员操作符协待值<W&&>)
    中 符号 协待(静转<W&&>(w));//全局协待
  异
    中 静转<W&&>(w);//自身转为`等待器`.
}

等待器可等待的返回对象.1,用专门带约束对象来实现挂起协程取协程返回值.2,灵活性和扩展性.可等待作为间接层,保存协待环境,可做更多事情.
等待器必须实现准备好等待(.1)/挂起等待(.2)/恢复等待(.3).
.1为假,就挂起协程,调用.2.如果.2返回真/空,就返回到调用者.返回,就执行协待下面语句..1为真,说明执行完协程,调用.3返回协程结果.

展示协待

构 任务{
  构 承诺类型{
    任务 取中对象(){中{协程句柄<任务::承诺类型>::从承诺(*本)};}
    从不挂起 初始挂起(){中{};}
    从不挂起 止挂起()无异{中{};}
    空 中空(){ 输出<<"取协程结果\n"; }//5
    空 未处理异常(){}
  };

  协程句柄<任务::承诺类型>句柄_;//外部控制.
};

任务 测试(){
  输出<<"创建协程\n";//1
  协待 总是挂起{};//在这里挂起了.
  输出<<"这是恢复点\n";//4
}
整 主(){
  动 t=测试();输出<<"现在返回调用者\n";//2
  输出<<"调用者恢复挂起协程\n";t.句柄_();//3
  输出<<"消灭协程\n";//6
}

可定义等待器,控制它的.1和.2来控制协程.
一般在.2中发起异步操作,此时协程是挂起的.返回调用者不阻塞.完成异步操作后,通过.3返回协程返回值,并恢复协程.结束协待,调用者拿到结果.

posted @   zjh6  阅读(16)  评论(0编辑  收藏  举报  
相关博文:
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现
点击右上角即可分享
微信分享提示