需要关于使用 libevent 动态更改定时器事件的建议

Need suggestion on dynamically changing timer event with libevent

我目前在我的 cpp (C++11) 程序中使用 libevent-2.0.5 来设置计时器事件。

代码逻辑如下:

    #include <stdio.h>
    #include <sys/time.h>
    #include <event.h>

    static void timer_cb(evutil_socket_t fd, short what, void *arg)
    {
        printf("something\n");
    }
  
    int main(int argc, char* argv[])
    {
        struct event *foo_timer;
        struct timeval one_sec = {.tv_sec = 1};
        struct event_base *base = event_base_new();

        // set up libvent timer to be executed every 1 second
        foo_timer = event_new(base, -1, EV_TIMEOUT|EV_PERSIST, timer_cb, args);
        event_add(foo_timer, &one_sec);

        // dispatch event
        event_base_dispatch(base);

    }

所以回调函数将每1秒执行一次

现在我的问题是,如何动态更改间隔?

例如,假设有另一个线程向这个定时器回调发送信号(不知道什么时候),例如设置一个 bool flag。 我要实现的是当回调发现flagtrue时,可以将时间间隔改为500ms。 这样定时器回调会在当前回调结束后500ms后执行。

我现在被困在这里,因为我发现的唯一可能方法是删除该计时器并创建(添加)一个新计时器。 喜欢:

    static void timer_cb(evutil_socket_t fd, short what, void *arg)
    {
        // do some checking 
        ...
        // create new event and remove the old one
        if (flag)
        {
            // create new timer
            bar_timer = event_new(base, -1, EV_TIMEOUT|EV_PERSIST, timer_cb, bar_timer);
            event_add(bar_timer, &five_100ms);
       
            // remove old one
            old_event = (struct event *)args;
            event_del(old_event);
        }
    }
  
    int main(int argc, char* argv[])
    {
        struct event *foo_timer;
         ...
        // set up libvent timer to be executed every 1s
        foo_timer = event_new(base, -1, EV_TIMEOUT|EV_PERSIST, timer_cb, foo_timer);
        event_add(foo_timer, &one_sec);
        // dispatch
         ...
    }

但是我在 event_del(old_event); 之后崩溃了 如果我放弃event_del,新事件好像不会触发

所以我需要一些关于如何实现我的目的的建议,或者我应该求助于其他一些库来实现它?

是的,唯一的方法是删除旧事件并创建一个超时为 500 毫秒的新事件。 检查指针(old_event)是否有效。

        old_event = (struct event *)args;
        event_del(old_event);

我的建议是使用 Libevent 2.1.1-alpha 版本的 event_self_cbarg。