libuv - 在没有多线程的情况下限制空闲事件的回调率而不阻塞线程

libuv - Limiting callback rate of idle event without blocking thread without multithreading

我正在使用 libsourcey,它使用 libuv 作为其底层 I/O 网络层。 一切都已设置好,似乎 运行(还没有测试任何东西,因为我只是在制作原型和试验)。但是,我要求在应用程序循环旁边(依赖于 libuv 循环的 libsourcey 附带的循环)也调用 "Idle function"。就像现在一样,它在每个周期都调用空闲 CB,这非常 CPU 消耗。我需要一种方法来限制 uv_idle_cb 的调用率而不阻塞调用线程,该线程与应用程序用来处理 I/O 数据的线程相同(不确定最后一条语句,请纠正我,如果我错了)。

idle 函数将管理应用程序的几个不同方面,它只需要在 1 秒内 运行 x 次。此外,一切都需要 运行 同一个线程(计划升级旧应用程序的网络基础设施,它 运行 完全是单线程的)。

这是我到目前为止的代码,其中还包括我在回调中休眠线程时所做的测试,但它会阻止所有内容,所以即使我设置的第二个空闲 cb 也具有与第一个相同的调用率.

struct TCPServers
{
    CTCPManager<scy::net::SSLSocket> ssl;
};

int counter = 0;
void idle_cb(uv_idle_t *handle)
{
    printf("Idle callback %d TID %d\n", counter, std::this_thread::get_id());

    counter++;

    std::this_thread::sleep_for(std::chrono::milliseconds(1000 / 25));
}

int counter2 = 0;
void idle_cb2(uv_idle_t *handle)
{
    printf("Idle callback2 %d TID %d\n", counter2, std::this_thread::get_id());

    counter2++;

    std::this_thread::sleep_for(std::chrono::milliseconds(1000 / 50));
}

class CApplication : public scy::Application
{
public:
    CApplication() : scy::Application(), m_uvIdleCallback(nullptr), m_bUseSSL(false)
    {}

    void start()
    {
        run();

        if (m_uvIdleCallback)
            uv_idle_start(&m_uvIdle, m_uvIdleCallback);

        if (m_uvIdleCallback2)
            uv_idle_start(&m_uvIdle2, m_uvIdleCallback2);
    }

    void stop()
    {
        scy::Application::stop();

        uv_idle_stop(&m_uvIdle);

        if (m_bUseSSL)
            scy::net::SSLManager::instance().shutdown();
    }

    void bindIdleEvent(uv_idle_cb cb)
    {
        m_uvIdleCallback = cb;
        uv_idle_init(loop, &m_uvIdle);
    }

    void bindIdleEvent2(uv_idle_cb cb)
    {
        m_uvIdleCallback2 = cb;
        uv_idle_init(loop, &m_uvIdle2);
    }

    void initSSL(const std::string& privateKeyFile = "", const std::string& certificateFile = "")
    {
        scy::net::SSLManager::instance().initNoVerifyServer(privateKeyFile, certificateFile);
        m_bUseSSL = true;
    }

private:
    uv_idle_t m_uvIdle;
    uv_idle_t m_uvIdle2;
    uv_idle_cb m_uvIdleCallback;
    uv_idle_cb m_uvIdleCallback2;
    bool m_bUseSSL;
};

int main()
{
    CApplication app;
    app.bindIdleEvent(idle_cb);
    app.bindIdleEvent2(idle_cb2);
    app.initSSL();
    app.start();

    TCPServers srvs;
    srvs.ssl.start("127.0.0.1", 9000);

    app.waitForShutdown([&](void*) {
        srvs.ssl.shutdown();
    });

    app.stop();

    system("PAUSE");
    return 0;
}

如果有人能提供帮助,在此先感谢。

通过使用 uv_timer_t 和 uv_timer_cb 解决了问题(还没有深入研究 libuv 的文档)。 CPU 使用率急剧下降,没有任何阻塞。