0

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

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

这是我到目前为止的代码,其中还包括我在回调中休眠线程所做的测试,但它会阻止所有内容,因此即使我设置的第二个空闲 cb 也与第一个空闲 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;
}

如果有人可以提供帮助,请提前致谢。

4

1 回答 1

1

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

于 2018-04-27T23:42:49.773 回答