libevent源码解析(一)核心数据结构

2024-09-05 05:32

本文主要是介绍libevent源码解析(一)核心数据结构,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一. 前言

  Libevent是一个轻量级的开源高性能网络库,对于学习网络编程、学习C语言等都有着很大的帮助。Libevent 有几个显著的亮点:事件驱动(event-driven),高性能;轻量级,专注于网络,不如ACE那么臃肿庞大;源代码相当精炼、易读;跨平台,支持Windows、Linux、BSD和Mac Os;支持多种I/O多路复用技术, epoll、poll、dev/poll、select和kqueue等;支持I/O,定时器和信号等事件;注册事件优先级。
  学习libevent库的时候,发现网上有很多优秀的文章,如张亮大神的libevent源码深度剖析系列。然而,他们的文章大多是多年以前的(甚至接近十年前),libevent中数据结构、框架结构、库的功能等等都有了不同程度的改动,所以在此自己边学习记录边解析新版的libevent的源码。我解析的是最新的2.18版本的libevent库。
  关于libevent的使用不用多说,网上一搜一大把。主要就是一个异步网络库,可以注册IO,信号等不同的事件。可以参考libevent源码剖析一到四篇的内容以及其他优秀博客加以了解。本文从核心数据结构入手,首先了解最基础的数据结构相关内容。

二. event

   Libevent是基于事件驱动(event-driven)的,从名字也可以看到event是整个库的核心。event就是Reactor框架中的事件处理程序组件;它提供了函数接口,供Reactor在事件发生时调用,以执行相应的事件处理,通常它会绑定一个有效的句柄。
   event结构在event.h中申明,在event_struct.h中定义。具体结构如下:

struct event {struct event_callback ev_evcallback;  /*回调函数结构体*//* for managing timeouts * 如果是timeout事件,它们是event在小根堆中的索引,* libevent使用小根堆来管理定时事件*/union {TAILQ_ENTRY(event) ev_next_with_common_timeout;int min_heap_idx;} ev_timeout_pos;evutil_socket_t ev_fd;	/*对于I/O事件,是绑定的文件描述符;对于signal事件,是绑定的信号*/short ev_events;	/*event关注的事件类型*/short ev_res;		/* 记录了当前激活事件的类型result passed to event callback */struct event_base *ev_base;	/*该事件所属的反应堆实例,这是一个event_base结构体*/union {/* used for io events */struct {LIST_ENTRY (event) ev_io_next;struct timeval ev_timeout;} ev_io;/* used by signal events */struct {LIST_ENTRY (event) ev_signal_next;short ev_ncalls;	/*事件就绪执行时,调用ev_callback的次数,通常为1*//* Allows deletes in callback 指针,通常指向ev_ncalls或者为NULL*/short *ev_pncalls;} ev_signal;} ev_;    /*针对不同类型的事件设置链表记录*/struct timeval ev_timeout;	/*timeout事件的超市值*/
};

这里用到了event_callback,该结构体如下所示:

struct event_callback {TAILQ_ENTRY(event_callback) evcb_active_next;short evcb_flags;ev_uint8_t evcb_pri;	/* smaller numbers are higher priority */ev_uint8_t evcb_closure;/* allows us to adopt for different types of events */union {void (*evcb_callback)(evutil_socket_t, short, void *);void (*evcb_selfcb)(struct event_callback *, void *);void (*evcb_evfinalize)(struct event *, void *);void (*evcb_cbfinalize)(struct event_callback *, void *);} evcb_cb_union;void *evcb_arg;
};

简单的说,该结构体定义了几种不同的而回调函数联合,用于不同的情况。
由上可见,event事件主要包括IO,信号,以及超时三种,在该头文件中定义了以下几种类型:

#define EVLIST_TIMEOUT	    0x01
#define EVLIST_INSERTED	    0x02
#define EVLIST_SIGNAL	    0x04
#define EVLIST_ACTIVE	    0x08
#define EVLIST_INTERNAL	    0x10
#define EVLIST_ACTIVE_LATER 0x20
#define EVLIST_FINALIZING   0x40
#define EVLIST_INIT	    0x80#define EVLIST_ALL          0xff

在上文中,我已经对event结构体中每个部分做了简要的说明,在event.h中,有详细的注释说明该结构体的作用以及函数接口,如下所示:

/*** @struct event** Structure to represent a single event.** An event can have some underlying condition it represents: a socket* becoming readable or writeable (or both), or a signal becoming raised.* (An event that represents no underlying condition is still useful: you* can use one to implement a timer, or to communicate between threads.)** Generally, you can create events with event_new(), then make them* pending with event_add().  As your event_base runs, it will run the* callbacks of an events whose conditions are triggered.  When you no* longer want the event, free it with event_free().** In more depth:** An event may be "pending" (one whose condition we are watching),* "active" (one whose condition has triggered and whose callback is about* to run), neither, or both.  Events come into existence via* event_assign() or event_new(), and are then neither active nor pending.** To make an event pending, pass it to event_add().  When doing so, you* can also set a timeout for the event.** Events become active during an event_base_loop() call when either their* condition has triggered, or when their timeout has elapsed.  You can* also activate an event manually using event_active().  The even_base* loop will run the callbacks of active events; after it has done so, it* marks them as no longer active.** You can make an event non-pending by passing it to event_del().  This* also makes the event non-active.** Events can be "persistent" or "non-persistent".  A non-persistent event* becomes non-pending as soon as it is triggered: thus, it only runs at* most once per call to event_add().  A persistent event remains pending* even when it becomes active: you'll need to event_del() it manually in* order to make it non-pending.  When a persistent event with a timeout* becomes active, its timeout is reset: this means you can use persistent* events to implement periodic timeouts.** This should be treated as an opaque structure; you should never read or* write any of its fields directly.  For backward compatibility with old* code, it is defined in the event2/event_struct.h header; including this* header may make your code incompatible with other versions of Libevent.** @see event_new(), event_free(), event_assign(), event_get_assignment(),*    event_add(), event_del(), event_active(), event_pending(),*    event_get_fd(), event_get_base(), event_get_events(),*    event_get_callback(), event_get_callback_arg(),*    event_priority_set()*/

三.event-base

   在event结构中,有一个指向event_base的指针,所以这里我们看看event_base结构体。该结构体定义于event_internal.h中,具体代码如下。

struct event_base {/** 事件库对应的函数指针 Function pointers and other data to describe this event_base's* backend. */const struct eventop *evsel;/** Pointer to backend-specific data. */void *evbase;/** 事件发生的变化存储在这里 List of changes to tell backend about at next dispatch.  Only used* by the O(1) backends. */struct event_changelist changelist;/** 针对信号事件的函数指针存放结构 Function pointers used to describe the backend that this event_base* uses for signals */const struct eventop *evsigsel;/** Data to implement the common signal handler code. */struct evsig_info sig;/*一些计数数据*//** Number of virtual events */int virtual_event_count;/** Maximum number of virtual events active */int virtual_event_count_max;/** Number of total events added to this event_base */int event_count;/** Maximum number of total events added to this event_base */int event_count_max;/** Number of total events active in this event_base */int event_count_active;/** Maximum number of total events active in this event_base */int event_count_active_max;/*事件循环控制参数*//** Set if we should terminate the loop once we're done processing* events. */int event_gotterm;/** Set if we should terminate the loop immediately */int event_break;/** Set if we should start a new instance of the loop immediately. */int event_continue;/** 优先级 The currently running priority of events */int event_running_priority;/** Set if we're running the event_base_loop function, to prevent* reentrant invocation. */int running_loop;/** Set to the number of deferred_cbs we've made 'active' in the* loop.  This is a hack to prevent starvation; it would be smarter* to just use event_config_set_max_dispatch_interval's max_callbacks* feature */int n_deferreds_queued;/* 此处用链表管理激活的事件 Active event management. *//** An array of nactivequeues queues for active event_callbacks (ones* that have triggered, and whose callbacks need to be called).  Low* priority numbers are more important, and stall higher ones.*/struct evcallback_list *activequeues;/** The length of the activequeues array */int nactivequeues;/** A list of event_callbacks that should become active the next time* we process events, but not this time. */struct evcallback_list active_later_queue;/* 超时控制 common timeout logic *//** An array of common_timeout_list* for all of the common timeout* values we know. */struct common_timeout_list **common_timeout_queues;/** The number of entries used in common_timeout_queues */int n_common_timeouts;/** The total size of common_timeout_queues. */int n_common_timeouts_allocated;/** IO事件 Mapping from file descriptors to enabled (added) events */struct event_io_map io;/** 信号事件 Mapping from signal numbers to enabled (added) events. */struct event_signal_map sigmap;/** 超时事件 Priority queue of events with timeouts. */struct min_heap timeheap;/** 时间戳记录 Stored timeval: used to avoid calling gettimeofday/clock_gettime* too often. */struct timeval tv_cache;struct evutil_monotonic_timer monotonic_timer;/** Difference between internal time (maybe from clock_gettime) and* gettimeofday. */struct timeval tv_clock_diff;/** Second in which we last updated tv_clock_diff, in monotonic time. */time_t last_updated_clock_diff;#ifndef EVENT__DISABLE_THREAD_SUPPORT/* threading support *//** The thread currently running the event_loop for this base */unsigned long th_owner_id;/** A lock to prevent conflicting accesses to this event_base */void *th_base_lock;/** A condition that gets signalled when we're done processing an* event with waiters on it. */void *current_event_cond;/** Number of threads blocking on current_event_cond. */int current_event_waiters;
#endif/** The event whose callback is executing right now */struct event_callback *current_event;#ifdef _WIN32/** IOCP support structure, if IOCP is enabled. */struct event_iocp_port *iocp;
#endif/** Flags that this base was configured with */enum event_base_config_flag flags;struct timeval max_dispatch_time;int max_dispatch_callbacks;int limit_callbacks_after_prio;/* Notify main thread to wake up break, etc. *//** True if the base already has a pending notify, and we don't need* to add any more. */int is_notify_pending;/** A socketpair used by some th_notify functions to wake up the main* thread. */evutil_socket_t th_notify_fd[2];/** An event used by some th_notify functions to wake up the main* thread. */struct event th_notify;/** A function used to wake up the main thread from another thread. */int (*th_notify_fn)(struct event_base *base);/** Saved seed for weak random number generator. Some backends use* this to produce fairness among sockets. Protected by th_base_lock. */struct evutil_weakrand_state weakrand_seed;/** List of event_onces that have not yet fired. */LIST_HEAD(once_event_list, event_once) once_events;};

  evsel和evbase这两个字段的设置可能会让人有些迷惑,这里你可以把evsel和evbase看作是类和静态函数的关系,比如添加事件时的调用行为:evsel->add(evbase, ev),实际执行操作的是evbase;这相当于class::add(instance, ev),instance就是class的一个对象实例。evsel指向了全局变量static const struct eventop *eventops[]中的一个;
前面也说过,libevent将系统提供的I/O demultiplex机制统一封装成了eventop结构;因此eventops[]包含了select、poll、kequeue和epoll等等其中的若干个全局实例对象。evbase实际上是一个eventop实例对象。eventop会根据当前编译环境选择可用的类型添加至数组中以供选择,其实现在编译过程中进行,对用户来说是完全封闭不可见的。具体说明在后文中会详细介绍。这里给出其中用到的event_top结构体如下:

/** Structure to define the backend of a given event_base. */
struct eventop {/** 名字 The name of this backend. */const char *name;/** 初始化 Function to set up an event_base to use this backend.  It should* create a new structure holding whatever information is needed to* run the backend, and return it.  The returned pointer will get* stored by event_init into the event_base.evbase field.  On failure,* this function should return NULL. */void *(*init)(struct event_base *);/** 注册 Enable reading/writing on a given fd or signal.  'events' will be* the events that we're trying to enable: one or more of EV_READ,* EV_WRITE, EV_SIGNAL, and EV_ET.  'old' will be those events that* were enabled on this fd previously.  'fdinfo' will be a structure* associated with the fd by the evmap; its size is defined by the* fdinfo field below.  It will be set to 0 the first time the fd is* added.  The function should return 0 on success and -1 on error.*/int (*add)(struct event_base *, evutil_socket_t fd, short old, short events, void *fdinfo);/** 删除 As "add", except 'events' contains the events we mean to disable. */int (*del)(struct event_base *, evutil_socket_t fd, short old, short events, void *fdinfo);/** 事件分发 Function to implement the core of an event loop.  It must see whichadded events are ready, and cause event_active to be called for eachactive event (usually via event_io_active or such).  It shouldreturn 0 on success and -1 on error.*/int (*dispatch)(struct event_base *, struct timeval *);/** 释放资源 Function to clean up and free our data from the event_base. */void (*dealloc)(struct event_base *);/** 标记位 Flag: set if we need to reinitialize the event base after we fork.*/int need_reinit;/**支持的特性,根据操作系统不同而不同 Bit-array of supported event_method_features that this backend can provide. */enum event_method_feature features;/** 额外信息记录 Length of the extra information we should record for each fd thathas one or more active events.  This information is recordedas part of the evmap entry for each fd, and passed as an argumentto the add and del functions above.*/size_t fdinfo_len;
};

  同样的,在event.h中有对其作用和调用函数的描述:

/*** Structure to hold information and state for a Libevent dispatch loop.** The event_base lies at the center of Libevent; every application will* have one.  It keeps track of all pending and active events, and* notifies your application of the active ones.** This is an opaque structure; you can allocate one using* event_base_new() or event_base_new_with_config().** @see event_base_new(), event_base_free(), event_base_loop(),*    event_base_new_with_config()*/

  由此我们可以看出,如果说event是核心数据结构,那么event_base就是在event之后,记录并统筹所有event的核心事件管理结构,即事件的处理框架。创建一个event_base对象也既是创建了一个新的libevent实例,程序需要通过调用event_init()(内部调用event_base_new函数执行具体操作)函数来创建,该函数同时还对新生成的libevent实例进行了初始化。该函数首先为event_base实例申请空间,然后初始化timer mini-heap,选择并初始化合适的系统I/O 的demultiplexer机制,初始化各事件链表;函数还检测了系统的时间设置,为后面的时间管理打下基础。

四.小结

  本文对libevent库的核心数据结构进行了解析,在下一篇中将会对本文中数据结构内提到的各个接口函数进行解析。

这篇关于libevent源码解析(一)核心数据结构的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1138085

相关文章

使用Java解析JSON数据并提取特定字段的实现步骤(以提取mailNo为例)

《使用Java解析JSON数据并提取特定字段的实现步骤(以提取mailNo为例)》在现代软件开发中,处理JSON数据是一项非常常见的任务,无论是从API接口获取数据,还是将数据存储为JSON格式,解析... 目录1. 背景介绍1.1 jsON简介1.2 实际案例2. 准备工作2.1 环境搭建2.1.1 添加

Java汇编源码如何查看环境搭建

《Java汇编源码如何查看环境搭建》:本文主要介绍如何在IntelliJIDEA开发环境中搭建字节码和汇编环境,以便更好地进行代码调优和JVM学习,首先,介绍了如何配置IntelliJIDEA以方... 目录一、简介二、在IDEA开发环境中搭建汇编环境2.1 在IDEA中搭建字节码查看环境2.1.1 搭建步

在C#中合并和解析相对路径方式

《在C#中合并和解析相对路径方式》Path类提供了几个用于操作文件路径的静态方法,其中包括Combine方法和GetFullPath方法,Combine方法将两个路径合并在一起,但不会解析包含相对元素... 目录C#合并和解析相对路径System.IO.Path类幸运的是总结C#合并和解析相对路径对于 C

Java解析JSON的六种方案

《Java解析JSON的六种方案》这篇文章介绍了6种JSON解析方案,包括Jackson、Gson、FastJSON、JsonPath、、手动解析,分别阐述了它们的功能特点、代码示例、高级功能、优缺点... 目录前言1. 使用 Jackson:业界标配功能特点代码示例高级功能优缺点2. 使用 Gson:轻量

Java如何接收并解析HL7协议数据

《Java如何接收并解析HL7协议数据》文章主要介绍了HL7协议及其在医疗行业中的应用,详细描述了如何配置环境、接收和解析数据,以及与前端进行交互的实现方法,文章还分享了使用7Edit工具进行调试的经... 目录一、前言二、正文1、环境配置2、数据接收:HL7Monitor3、数据解析:HL7Busines

python解析HTML并提取span标签中的文本

《python解析HTML并提取span标签中的文本》在网页开发和数据抓取过程中,我们经常需要从HTML页面中提取信息,尤其是span元素中的文本,span标签是一个行内元素,通常用于包装一小段文本或... 目录一、安装相关依赖二、html 页面结构三、使用 BeautifulSoup javascript

网页解析 lxml 库--实战

lxml库使用流程 lxml 是 Python 的第三方解析库,完全使用 Python 语言编写,它对 XPath表达式提供了良好的支 持,因此能够了高效地解析 HTML/XML 文档。本节讲解如何通过 lxml 库解析 HTML 文档。 pip install lxml lxm| 库提供了一个 etree 模块,该模块专门用来解析 HTML/XML 文档,下面来介绍一下 lxml 库

Andrej Karpathy最新采访:认知核心模型10亿参数就够了,AI会打破教育不公的僵局

夕小瑶科技说 原创  作者 | 海野 AI圈子的红人,AI大神Andrej Karpathy,曾是OpenAI联合创始人之一,特斯拉AI总监。上一次的动态是官宣创办一家名为 Eureka Labs 的人工智能+教育公司 ,宣布将长期致力于AI原生教育。 近日,Andrej Karpathy接受了No Priors(投资博客)的采访,与硅谷知名投资人 Sara Guo 和 Elad G

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟 开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚 第一站:海量资源,应有尽有 走进“智听

【数据结构】——原来排序算法搞懂这些就行,轻松拿捏

前言:快速排序的实现最重要的是找基准值,下面让我们来了解如何实现找基准值 基准值的注释:在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。 在此我们采用三数取中法,也就是取左端、中间、右端三个数,然后进行排序,将中间数作为枢纽值。 快速排序实现主框架: //快速排序 void QuickSort(int* arr, int left, int rig