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

Posted

技术标签:

【中文标题】需要关于使用 libevent 动态更改计时器事件的建议【英文标题】:Need suggestion on dynamically changing timer event with libevent 【发布时间】:2019-05-17 06:54:22 【问题描述】:

我目前在我的 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秒执行一次。

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

例如,假设有另一个线程向这个定时器回调发送信号(不知道什么时候),例如设置一个布尔值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,新事件似乎没有触发。

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

【问题讨论】:

【参考方案1】:

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

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

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

【讨论】:

以上是关于需要关于使用 libevent 动态更改计时器事件的建议的主要内容,如果未能解决你的问题,请参考以下文章

libevent源码分析-介绍安装使用

libevent学习笔记 —— 第一个程序:计时器

libevent

一旦发生另一个事件,我的 libevent 计时器就会停止……这正常吗?

libevent总结(上)

libevent激活事件