c++ - 在协程执行之间强制休眠

标签 c++ asynchronous boost coroutine boost-coroutine

我目前有一些同步 C++ 代码,它们以顺序方式执行一些耗时的任务。我正在考虑使用 Boost 协程将其重构为并行任务。

任务的核心是调用外部库,该库提供同步 API(它当前正在使用)或异步 API,必须定期轮询以进行操作并确定它是否已完成。该 API 不是线程安全的,但如果使用异步版本,只要不同时调用轮询 API 本身,它就可以处理并发发生的多个请求。

(由于我不想进行任何线程转移,看来我应该直接使用协程 API 而不是使用 ASIO 包装器或类似工具,但我愿意被说服否则。我也在使用 Boost 1.55目前,因此 Fiber 库不可用;但对于这种用法来说似乎也有点过分了。)

目前我对协程的主要问题是我不确定如何对轮询实现限制——假设我有 100 个任务要并行运行;我希望它对所有 100 个任务执行一次轮询,然后让线程休眠指定的时间。有些任务可能比其他任务完成得更快,因此稍后它可能仍会在每个循环中轮询 40 个任务,然后休眠相同的时间。我不希望它在没有 sleep 干预的情况下连续两次轮询同一个任务。

最佳答案

好的,事实证明比我想象的要容易得多。这是我到目前为止的结果:

typedef boost::coroutines::coroutine<void>::push_type coroutine;
typedef boost::coroutines::coroutine<void>::pull_type yield_context;
typedef std::vector<coroutine> coroutine_list;

bool run_once_all(coroutine_list& coros)
{
    bool pending = false;
    for (auto& coro : coros)
    {
        if (coro)
        {
            coro();
            pending = pending || !coro.empty();
        }
    }
    return pending;
}

void run_all(coroutine_list& coros,
             const boost::chrono::nanoseconds& idle_ns)
{
    while (run_once_all(coros))
    {
        boost::this_thread::sleep_for(idle_ns);
    }
}

void run_all(coroutine_list& coros, yield_context& yield)
{
    while (run_once_all(coros))
    {
        yield();
    }
}

我们的想法是创建一个 coroutine_list,然后是带有签名 void (yield_context&)emplace_back lambda 函数,然后是 run_all。第一个用于顶层,第二个用于嵌套协程。

似乎有效,虽然我还没有给它一个主要的锻炼。实际上有点惊讶图书馆里没有这样的东西。

关于c++ - 在协程执行之间强制休眠,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36538017/

相关文章:

c++ - 丑陋的数字UVA

c# - boost proto vs c# 表达式树

c++ - Boost 安装和库路径

c++ - 如果 T 可以为空,如何在 boost 中安全地获取 sizeof(T)?

c++ - C++ 中的右值引用和一个简单的例子

c++ - 将值放入二维数组

javascript - AngularJS 的异步特性与摘要/应用

javascript - 有条件地调用一个 promise (或不调用),但将任一结果返回给另一个 promise

asynchronous - 加载 Google Places 自动完成异步 Angular 2

c++ - 在字符串中输出数据