如何打破libevent的调度循环

时间:2015-03-23 20:02:42

标签: c++ multithreading libevent

我使用nice libevent库在一个单独的线程中实现了一个小型的Web服务器。 Web服务器运行event_base_dispatch()来处理所有事件。我需要的是一种从主线程中打破这个调度循环的方法。

归结为以下C ++代码:

#include <stdlib.h>
#include <signal.h>
#include <thread>
#include <evhttp.h>

struct event_base *eb;
std::thread t;

static volatile sig_atomic_t bailout = false;

void my_signal_handler(int) {
    bailout = true;
}

void onRequest(evhttp_request *req, void *) {
    struct evbuffer *OutBuf = evhttp_request_get_output_buffer(req);
    evbuffer_add_printf(OutBuf, "<html><body>Testing 1-2-3</body></html>");
    evhttp_send_reply(req, HTTP_OK, "OK", OutBuf);
}

void dispatch() {
    eb = event_base_new();
    struct evhttp *http = evhttp_new(eb);
    evhttp_set_gencb(http, &onRequest, NULL);
    evhttp_bind_socket_with_handle(http, "0.0.0.0", 5555);
    event_base_dispatch(eb);
}

int main() {

    struct sigaction sigIntHandler;
    sigIntHandler.sa_handler = my_signal_handler;
    sigemptyset(&sigIntHandler.sa_mask);
    sigIntHandler.sa_flags = 0;
    sigaction(SIGINT, &sigIntHandler, NULL);

    t = std::thread { &dispatch };

    while ( ! bailout ) {
        std::this_thread::sleep_for(std::chrono::seconds(1));
    }

    event_base_loopexit(eb, NULL);

    t.join();
}

行为是,如果您运行程序,请求页面,通过Ctrl-C中止程序,event_base_dispatch()将继续运行,直到您获取另一个网页。只有这样循环才会中止,程序就会终止。

3 个答案:

答案 0 :(得分:1)

如果在运行任何更多回调之前需要退出事件循环,请使用event_base_break()

http://www.wangafu.net/~nickm/libevent-book/Ref3_eventloop.html

答案 1 :(得分:1)

最后,我通过添加一个基本上轮询救助变量的计时器实现了分出循环:

struct event *ev;
const struct timeval one_sec = { 1, 0 }; // sec, usec

void cb_timer_func(evutil_socket_t, short, void) {

  if ( bailout ) {
    event_base_loopbreak(eb);
  }
  else if ( ! evtimer_pending(ev, NULL) ) {
    evtimer_del(ev);
    evtimer_add(ev, &one_sec);
  }
}

在调度循环中,我补充道:

ev = evtimer_new(eb, cb_timer_func);
evtimer_add(ev, &one_sec);

答案 2 :(得分:0)

另一种方法是将event_base_loopbreak()放入信号处理程序。