libevent学习笔记十一:libevent支持I/O多路复用技术

libevent学习笔记十一:libevent支持I/O多路复用技术

     Libevent的核心是事件驱动、同步非阻塞,为了达到这一目标,必须采用系统提供的I/O多路复用技术,而这些在Windows、Linux、Unix等不同平台上却各有不同,如何能够跨平台、跨操作系统提供优雅而统一的支持方式,是首要关键要解决的问题,本节的内容就是就如何实现做的分析。

1、统一模式的关键

       Libevent支持多种I/O多路复用技术的关键就在于结构体eventop,这个结构体它的成员是一系列的函数指针, 定义在event-internal.h文件中:

 struct eventop {
     const char *name;
     void *(*init)(struct event_base *);                              // 初始化
     int (*add)(void *, struct event *);                              // 注册事件
     int (*del)(void *, struct event *);                              // 删除事件
     int (*dispatch)(struct event_base *, void *, struct timeval *);  // 事件分发
     void (*dealloc)(struct event_base *, void *);                    // 注销,释放资源
     /* set if we need to reinitialize the event base */
     int need_reinit;
 };

     在上面代码中,我们可以看到这个结构体中,包含了五个函数。在libevent中,每种I/O demultiplex机制的实现都必须提供这五个函数接口,来完成自身的初始化、销毁释放;对事件的注册、注销和分发。

    比如对于epoll,libevent实现了5个对应的接口函数,并在初始化时并将eventop的5个函数指针指向这5个函数,那么程序就可以使用epoll作为I/O demultiplex机制了。

 

2、设置I/O demultiplex机制

     Libevent把所有支持的I/O demultiplex机制存储在一个全局静态数组eventops中,并在初始化时选择使用何种机制,数组内容根据优先级顺序声明如下:

 /* In order of preference */
 static const struct eventop *eventops[] = {
 #ifdef HAVE_EVENT_PORTS
     &evportops,
 #endif
 #ifdef HAVE_WORKING_KQUEUE
     &kqops,
 #endif
 #ifdef HAVE_EPOLL
     &epollops,
 #endif
 #ifdef HAVE_DEVPOLL
     &devpollops,
 #endif
 #ifdef HAVE_POLL
     &pollops,
 #endif
 #ifdef HAVE_SELECT
     &selectops,
 #endif
 #ifdef WIN32
     &win32ops,
 #endif
     NULL
 }; 

 然后libevent根据系统配置和编译选项决定使用哪一种I/O demultiplex机制,这段代码在函数event_base_new()中:

base->evbase = NULL;
    for (i = 0; eventops[i] && !base->evbase; i++) {
         base->evsel = eventops[i];
         base->evbase = base->evsel->init(base);
     } 

可以看出,libevent在编译阶段选择系统的I/O demultiplex机制,而不支持在运行阶段根据配置再次选择。
 以Linux下面的epoll为例,实现在源文件epoll.c中,eventops对象epollops定义如下:

const struct eventop epollops = {
    "epoll",
    epoll_init,
    epoll_add,
    epoll_del,
    epoll_dispatch,
    epoll_dealloc,
    1 /* need reinit */
};

变量epollops中的函数指针具体声明如下,注意到其返回值和参数都和eventop中的定义严格一致,这是函数指针的语法限制。

static void *epoll_init    (struct event_base *);
static int epoll_add    (void *, struct event *);
static int epoll_del    (void *, struct event *);
static int epoll_dispatch(struct event_base *, void *, struct timeval *);
static void epoll_dealloc    (struct event_base *, void *);

      那么如果选择的是epoll,那么调用结构体eventop的init和dispatch函数指针时,实际调用的函数就是epoll的初始化函数epoll_init()和事件分发函数epoll_dispatch()了;关于epoll的原理可以参见网上介绍epoll的文章。

     这里需要强调的是信息隐藏,这个原理实际可以追踪到C++语言提供了虚函数来实现多态,而在C语言中,是通过函数指针实现的。对于各类函数指针的详细说明可以参见网上的关于这方面的文章。对于epoll的隐藏,epollops以及epoll的各种函数都直接定义在了epoll.c源文件中,对外都是不可见的。对于libevent的使用者而言,完全不会知道它们的存在,对epoll的使用也是通过eventop来完成的,达到了信息隐藏的目的。

3、总结

         支持多种I/O demultiplex机制的方法其实挺简单的,借助于函数指针来实现。对源代码的分析也可以看出,Libevent是在编译阶段选择系统的I/O demultiplex机制的,而不支持在运行阶段根据配置再次选择。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

jyl_sh

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值