gpt4 book ai didi

c++ - 如何用 C++20 协程说 Hello World?

转载 作者:塔克拉玛干 更新时间:2023-11-03 02:14:09 26 4
gpt4 key购买 nike

出于学习目的,我尝试使用 C++20 协程制作过于复杂的“Hello World”程序:

HelloWorldMessage sayHelloToWorld()
{
co_yield "Hello";
co_yield " ";
co_yield "World";
co_yield "!";
}

int main()
{
for (auto w : sayHelloToWorld())
{
std::cout << w;
}
}

为了准备这样的 HelloWorldMessage 生成器,我主要基于最新的 clang 警告消息和 uncomplete cppreference page还有这个example .

所以我的结果如下。这里缺少什么?因为,我没有说你好,而是遇到了段错误:

参见 link :

struct HelloWorldState
{
const char* currentWord = "<not value yet>";
bool finalWord = false;
};

struct HelloWorldPromise
{
HelloWorldState state;
std::experimental::suspend_always initial_suspend() const noexcept { return {}; }
std::experimental::suspend_always final_suspend() const noexcept { return {}; }

std::experimental::suspend_always yield_value(const char* word) noexcept
{
state.currentWord = word;
return {};
}

std::experimental::suspend_always return_void() noexcept
{
state.finalWord = true;
return {};
}

auto& get_return_object() noexcept
{
return *this;
}

void unhandled_exception()
{
state.finalWord = true;
throw;
}
};

struct HelloWorldMessage
{
using promise_type = HelloWorldPromise;
using promise_handle = std::experimental::coroutine_handle<promise_type>;

struct Iter
{
promise_handle handle = nullptr;
HelloWorldState state;

using iterator_category = std::input_iterator_tag;
using value_type = const char*;
using difference_type = ptrdiff_t;
using pointer = value_type const *;
using reference = value_type const &;

reference operator * () const { assert(handle); return state.currentWord; }
pointer operator -> () const { return std::addressof(operator*()); }

bool operator == (const Iter& other) { return handle == other.handle; }
bool operator != (const Iter& other) { return !(*this == other); }

Iter() = default;
Iter(promise_handle handle)
: handle(handle)
{
assert(handle);
next();
}

Iter& operator ++()
{
if (!handle)
return *this;
if (state.finalWord)
{
handle = nullptr;
return *this;
}
next();
return *this;
}

void next()
{
try {
handle.resume();
state = handle.promise().state;
} catch (...) {
std::cerr << "@%$#@%#@$% \n";
}
}

};

promise_handle handle = nullptr;
HelloWorldMessage(promise_type& promise) : handle(promise_handle::from_promise(promise)) {}

Iter begin() const { assert(handle); return {handle}; }
Iter end() const { return {}; }
};

也许 clang 还没有准备好?

最佳答案

一些错误:

首先 - promise 应返回生成器对象,而不是对自身的引用。所以正确的做法是:

struct HelloWorldPromise
{
...
auto get_return_object();
...
};

struct HelloWorldMessage
{
...
};

auto HelloWorldPromise::get_return_object()
{
return HelloWorldMessage(*this);
}

下一步 - 终止并返回 void 可以简化为:

void return_void() noexcept
{}
void unhandled_exception()
{
std::terminate();
}

下一步 - 在迭代器中 - 我们将依赖 handle.done - 所以不需要 state.finalWord。完整的迭代器源是:

struct Iter
{
promise_handle handle = nullptr;
HelloWorldState state;

reference operator * () const { return state.currentWord; }
pointer operator -> () const { return std::addressof(operator*()); }

bool operator == (const Iter& other) const { return !handle == !other.handle; }
bool operator != (const Iter& other) const { return !(*this == other); }

Iter() = default;
Iter(promise_handle handle)
: handle(handle)
{
next();
}

Iter& operator ++()
{
if (!handle)
return *this;
next();
return *this;
}

void next()
{
if (!handle)
return;
try {
handle.resume();
if (!handle.done())
state = handle.promise().state;
else {
handle = nullptr;
}
} catch (...) {
std::cerr << "@%$#@%#@$% \n";
}
}

};

以及完整的工作示例 here .

我的大部分更正都来自这个 2018/n4736.pdf .

关于c++ - 如何用 C++20 协程说 Hello World?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56185517/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com