libuv - ограничение скорости обратного вызова для бездействующего события без блокировки потока без многопоточности - PullRequest
0 голосов
/ 28 апреля 2018

Я использую libsourcey, которая использует libuv в качестве базового сетевого уровня ввода / вывода. Все настроено и, кажется, работает (еще ничего не тестировал, так как я только создаю прототипы и экспериментирую). Тем не менее, я требую, чтобы рядом с циклом приложения (тот, который поставляется с libsourcey, который опирается на цикл libuv), также вызывается «Idle function». Как и сейчас, он вызывает Idle CB на каждом цикле, который сильно загружает процессор. Мне нужен способ ограничить скорость вызова uv_idle_cb, не блокируя вызывающий поток, который является тем же, который приложение использует для обработки данных ввода / вывода (не уверен насчет этого последнего оператора, поправьте меня, если я ошибаюсь).

Функция бездействия будет управлять несколькими различными аспектами приложения и должна запускаться только x раз в течение 1 секунды. Кроме того, все должно работать в одном и том же потоке (планируется обновить сетевую инфраструктуру старого приложения, которая работает полностью однопоточным).

Это код, который у меня есть до сих пор, который также включает в себя тест, который я провел со спящим потоком в обратном вызове, но он блокирует все, так что даже второй установленный мной неактивный 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;
}

Заранее спасибо, если кто-нибудь может помочь.

1 Ответ

0 голосов
/ 28 апреля 2018

Решил проблему, используя uv_timer_t и uv_timer_cb (еще не копался в документации libuv). Загрузка ЦП резко снизилась, и ничто не блокируется.

...