基于C++ Coroutines提案 ‘Stackless Resumable Functions’编写的协程库
Nevar pievienot vairāk kā 25 tēmas Tēmai ir jāsākas ar burtu vai ciparu, tā var saturēt domu zīmes ('-') un var būt līdz 35 simboliem gara.

test_async_cb.cpp 2.7KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104
  1. #include <chrono>
  2. #include <iostream>
  3. #include <string>
  4. #include <thread>
  5. #include "librf.h"
  6. using namespace resumef;
  7. template<class _Ctype>
  8. static void callback_get_long(int64_t val, _Ctype&& cb)
  9. {
  10. using namespace std::chrono;
  11. std::thread([val, cb = std::forward<_Ctype>(cb)]
  12. {
  13. std::this_thread::sleep_for(500ms);
  14. cb(val * val);
  15. }).detach();
  16. }
  17. //这种情况下,没有生成 frame-context,因此,并没有promise_type被内嵌在frame-context里
  18. static future_t<int64_t> async_get_long(int64_t val)
  19. {
  20. awaitable_t<int64_t> awaitable;
  21. callback_get_long(val, [awaitable](int64_t val)
  22. {
  23. awaitable.set_value(val);
  24. });
  25. return awaitable.get_future();
  26. }
  27. static future_t<int64_t> wait_get_long(int64_t val)
  28. {
  29. co_return co_await async_get_long(val);
  30. }
  31. //这种情况下,会生成对应的 frame-context,一个promise_type被内嵌在frame-context里
  32. static future_t<int64_t> resumable_get_long(int64_t val)
  33. {
  34. std::cout << val << std::endl;
  35. val = co_await wait_get_long(val);
  36. std::cout << val << std::endl;
  37. val = co_await wait_get_long(val);
  38. std::cout << val << std::endl;
  39. val = co_await wait_get_long(val);
  40. std::cout << val << std::endl;
  41. co_return val;
  42. }
  43. static future_t<int64_t> loop_get_long(int64_t val)
  44. {
  45. std::cout << val << std::endl;
  46. for (int i = 0; i < 5; ++i)
  47. {
  48. val = co_await async_get_long(val);
  49. std::cout << val << std::endl;
  50. }
  51. co_return val;
  52. }
  53. static future_t<std::string&> async_get_string(std::string & ref_string)
  54. {
  55. awaitable_t<std::string&> awaitable;
  56. callback_get_long(std::stoi(ref_string), [awaitable, &ref_string](int64_t val)
  57. {
  58. ref_string = std::to_string(val);
  59. awaitable.set_value(ref_string);
  60. });
  61. return awaitable.get_future();
  62. }
  63. static future_t<std::string&> resumable_get_string(std::string& val)
  64. {
  65. std::cout << val << std::endl;
  66. val = co_await async_get_string(val);
  67. std::cout << val << std::endl;
  68. val = co_await async_get_string(val);
  69. std::cout << val << std::endl;
  70. val = co_await async_get_string(val);
  71. std::cout << val << std::endl;
  72. co_return static_cast<std::string&>(val);
  73. }
  74. void resumable_main_cb()
  75. {
  76. //由于使用者可能不能明确的区分是resume function返回的awaitor还是awaitable function返回的awaitor
  77. //导致均有可能加入到协程里去调度。
  78. //所以,协程调度器应该需要能处理这种情况。
  79. go async_get_long(3);
  80. this_scheduler()->run_until_notask();
  81. std::string ref_string{"2"};
  82. go resumable_get_string(ref_string);
  83. this_scheduler()->run_until_notask();
  84. GO
  85. {
  86. auto val = co_await resumable_get_long(2);
  87. std::cout << "GO:" << val << std::endl;
  88. };
  89. go loop_get_long(3);
  90. this_scheduler()->run_until_notask();
  91. }