- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
_-6ren">
以下代码不符合我的预期。
#include <iostream>
#include <coroutine>
#include <vector>
struct symmetic_awaitable
{
std::coroutine_handle<> _next_h;
symmetic_awaitable(std::coroutine_handle<> h) : _next_h(h) {}
constexpr bool await_ready() const noexcept { return false; }
std::coroutine_handle<> await_suspend(std::coroutine_handle<>) const noexcept
{
return _next_h;
}
constexpr void await_resume() const noexcept {}
};
struct return_object : std::coroutine_handle<>
{
struct promise_type
{
return_object get_return_object()
{
return std::coroutine_handle<promise_type>::from_promise(*this);
}
std::suspend_always initial_suspend() noexcept { return {}; }
std::suspend_always final_suspend() noexcept { return {}; }
void return_void() {}
void unhandled_exception() {}
};
return_object(std::coroutine_handle<promise_type> h) : std::coroutine_handle<>(h) {}
};
std::vector<return_object> co_list;
return_object fa()
{
std::cout << "fa1" << std::endl;
co_await symmetic_awaitable(co_list[1]);
std::cout << "fa2" << std::endl;
co_return;
}
return_object fb()
{
std::cout << "fb1" << std::endl;
co_await symmetic_awaitable(co_list[2]);
std::cout << "fb2" << std::endl;
co_await std::suspend_always{};
std::cout << "fb3" << std::endl;
co_return;
}
return_object fc()
{
std::cout << "fc1" << std::endl;
co_await symmetic_awaitable(co_list[1]);
std::cout << "fc2" << std::endl;
co_return;
}
int main()
{
auto a = fa();
auto b = fb();
auto c = fc();
co_list.push_back(a);
co_list.push_back(b);
co_list.push_back(c);
a.resume();
std::cout << "end" << std::endl;
a.destroy();
b.destroy();
c.destroy();
}
我认为输出会是
fa1
fb1
fc1
fb2
fa2
end
但实际输出是
fa1
fb1
fc1
fb2
end
然后我将所有 co_await symmetic_awaitable(co_list[i])
替换为 co_list[i].resume
。输出很奇怪
fa1
fb1
fc1
fb1
fc1
..... // the following is infinite loop of "fb1 fc1"
.....
.....
C++20的协程隐藏了太多的细节,除非我都清楚,否则代码无法像我期望的那样正常运行。
以下是我阅读 cppreference 后的问题:
a调用b.resume()
,那么a是b的恢复者还是调用者?
a调用b.resume()
,那么a是挂起还是运行? a 通过 co_await
恢复 b,然后 a 挂起或运行?
最佳答案
如果一个函数是协程,它只能通过以下方式之一暂停:
co_await
表达式(或等价物,如 co_yield
)时。co_return
时,如果 promise 最终挂起。没有别的可以导致协程挂起。 “coroutine”中的“co”代表协作式多任务处理。这意味着除非功能涉及合作,否则不会进行多任务处理。明确地。
您关于期望代码如何工作的假设似乎表明您相信协程具有某种执行堆栈。当 await_suspend
被调用时,当前协程被放入一个堆栈,当您返回的协程句柄以某种方式完成时,该堆栈将被弹出。因此,当您调用 co_await std::suspend_always{};
时,这将恢复之前挂起的协程。
这些都不是真的。除非你自己 build 那台机器。
协程系统完全且只您告诉它做什么。
调用 a.resume()
后的调用堆栈如下所示:
main()
fa()
当fa
暂停和恢复fb
时,它现在看起来像这样:
main()
fb()
fa
已消失。你暂停了它。它不再在调用堆栈上。只有当您明确要求恢复时,它才会恢复。
如果你想让fa
挂起到fb
中意味着fa
会在fb
结束后继续,那么您必须将其构建到您的协程机器中。它不只是发生;实现它是您的责任。
您的 await_suspend
代码需要获取给定的句柄(指的是 fa
)并将其存储在一个地方,当 fb
完成后,它可以恢复 fa
。这通常在 fb
的 promise 对象中,以便 final_suspend
可以恢复它(通常传递 fb
生成的数据)。请记住:无论 promise 的 final_suspend
返回什么,最终挂起点都会 co_await
,因此您可以只返回要恢复的协程的句柄。
What's the defference of "caller" and "resumer" ?
我不知道那是什么意思。我怀疑您是在问 co_await
ing 和直接调用 coroutine_handle::resume
函数之间的区别。
如前所述,在初始和最终挂起点之外,仅 co_await
(或等效)表达式可以导致协程挂起。在句柄上调用 resume
就像调用函数的中间部分一样。它的工作方式与任何其他函数调用一样;它进入堆栈等等。
让 co_await
恢复协程是不同的。当 await_suspend
返回协程句柄时,这会用调用堆栈上恢复的协程替换您的协程。这就是暂停协程的全部意义所在。
关于c++ - "co_await other_co"和 "other_co.resume"有什么区别?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/72504525/
平时很少在jquery中用到this。查看代码时发现用到了,就调试出this的值,心想原来如此。还是挺有用的。这里总结一下this与$(this)的区别和使用。 $(this)生成的是什么?
使用单例类和应用程序范围的托管 bean 来保存应用程序数据有区别吗? 我需要查找某些 JNDI 资源,例如远程 bean 接口(interface),因此我为自己编写了一个单例来缓存我的引用并且只允
如果您仔细查看包含的图片,您会注意到您可以使用 Eclipse IDE 重构 Groovy 代码并将方法转换为闭包,反之亦然。那么,闭包到底是什么,它与方法有什么不同呢?有人可以举一个使用闭包的好例子
vagrant box repackage有什么区别( docs ) 和 vagrant package ( docs )? 我意识到 vagrant package仅适用于 VirtualBox 提
我想看看是否有人可以解释为什么以下代码适用于 valueOf 但不适用于其他代码。 import java.math.BigDecimal; public class Change { publ
这个问题已经有答案了: 已关闭12 年前。 Possible Duplicates: What is Closures/Lambda in PHP or Javascript in layman te
This question already has answers here: Vagrant, Docker, Puppet, Chef (3个答案) 2年前关闭。 docker和chef有什么共同
以下代码在95%的机器上产生相同的输出,但是在几台机器上却有所不同。在 Debug模式下,输出: Changing from New to Fin OK 但在 Release模式下: Changing
////Creating Object var Obj; // init Object Obj= {}; 它们之间有什么区别两个? 有没有可能把它变成一个单行? 这样使用有什么好处吗?
我想找出定时器服务之间的区别。我应该使用哪个以及何时使用。我正在使用 Jboss 应用服务器。 1) java.ejb.Schedule。 @Schedule注解或配置自xml。 2) javax.e
我发现在 C++ 中可以通过三种不同的方式将对象传递给函数。假设我的类(class)是这样的: class Test { int i; public: Test(int x);
有什么区别。 public class Test { public static void main(String args[]) { String toBeCast = "c
如果我有一列,设置为主索引,设置为INT。 如果我不将其设置为自动递增,而只是将唯一的随机整数插入其中,与自动递增相比,这是否会减慢 future 的查询速度? 如果我在主索引和唯一索引为 INT 的
这两种日期格式有什么区别。第一个给出实际时间,第二个给出时间购买添加时区偏移值。 NSDateFormatter * dateFormatter = [[NSDateFormatter alloc]
如果有一个函数,请说foo: function foo() { console.log('bar'); } 那么在 JavaScript 中,从另一个函数调用一个函数有什么区别,如下所示: f
关闭。这个问题是opinion-based 。目前不接受答案。 想要改进这个问题吗?更新问题,以便 editing this post 可以用事实和引文来回答它。 . 已关闭 4 年前。 Improv
代码是什么: class Time { private: int hours; int minutes; int seconds; pu
我知道这是非常基本的,但有人介意解释一下这两个数组声明之间的区别吗: #include array myints; ...和: int myints[5]; ...以及为什么 myints.size
我学会了如何根据 http://reference.sitepoint.com/css/specificity 计算 css 特异性但是,基于this reference,我不明白伪类(来自c)和伪元
为什么在运行 2) 时会出现额外的空行?对我来说 1 就像 2。那么为什么 2) 中的额外行? 1) export p1=$(cd $(dirname $0) && pwd) #
我是一名优秀的程序员,十分优秀!