有什么优雅的方法可以让 c++20 协程句柄到达外部?

any elegant way to get c++20 coroutine handle to the outside?

我们知道 c++20 coroutine 在使用 co_await awaitable 后挂起和 return caller,如果我们想恢复这个协程,唯一的方法是从 awaitable 对象中获取协程句柄函数 void await_suspend(std::coroutine_handle<> cont) const 然后使用 handle.resume() 恢复协程,有没有办法从协程外部获取协程句柄或者有更好的方法恢复协程?

从广义上讲,您不应该想要从协程机器外部获取句柄。如果这样做,很可能您没有按照预期的方式使用协程。

C++20 中设计的协同程序的思想是不可见。也就是说,从协程之外的代码来看没有人应该知道它是作为协程实现的。只是returns某种类型的函数,表示一个未来可以得到的值;它是如何做到这一点的是一个实现细节。

“恢复协程”意味着告诉协程函数返回的未来对象产生值。这应该只是对 future 对象的常规 API 调用。可以访问 promise 的是 future 对象,也是可以访问协程句柄以恢复它的 promise。

此外,除非协程是生成器,否则直接恢复协程执行的代码应该是协程 co_await 运行的“可等待”代码。也就是说,有一些进程会产生一个值,你 co_await 这个进程 returns 的未来,一旦值准备好,这个进程就会恢复你的协程。如果其他人需要 co_await 您的协程产生的值,那很好。您的 promise/future 机器将在您的协程完成时恢复它们,并为它们提供您的承诺值。

这就是 co_await 协程的恢复方式。此进程之外的代码不需要句柄,也不应获得句柄。

无论如何,除非协程的promise/future机制明确地给你一个API来提取句柄,否则你无法得到句柄。

嗯,我参考c++23std::generator找到了一个更好的从调用者恢复协程的方法,如下所示:

struct Result{
  //add
  Result(promise_type* obj):promise_type_ptr(obj){}
  //add
  void resume(){
    promise_type_ptr->resume();
  }

  struct promise_type {
    // mod
    Result get_return_object() { 
        return Reuslt(this);
    }
    
    // add
    void resume(){
        coroutine_handle<promise_type>::from_promise(*this).resume();
    }
    
    std::suspend_never initial_suspend() { return {}; }
    std::suspend_never final_suspend() noexcept { return {}; }
    void unhandled_exception() {}
    
    suspend_aways yield_value(){}
    void return_void() {}   
    Result return_value(const Result& res){ return res;}
  };
  
  // add
  promise_type *promise_type_ptr;
};

我们可以使用coroutine_handle<promise_type>::from_promise(*this);从promise_type对象指针获取协程句柄,因此我们可以在协程return结果中为调用者创建一个函数resume(),并且如下使用

auto result = CoroutineFunction();
result.resume();