Libevent库的学习

article/2025/9/10 13:08:12

目录

Libevent 概述

Libevent 使用模型

使用Libevent的基本流程:

libevent 的核心,event 事件

1. 创建一个事件event

2. 释放event_free

3. 注册event

4. 信号事件 

5. 销毁event_base 

Libevent 结构图

使用libevent库去实现tcp服务器


Libevent 概述

Libevent 是开源社区的一款高性能的 I/O 框架库,使用 Libevent 的著名案例有:高性能 的分布式内存对象缓存软件 memcached,Google 浏览器 Chromium 的 Linux 版本。作为一个 I/O 框架库,

Libevent 具有如下特点:

◼ 跨平台支持。 Libevent 支持 Linux、Unix 和 Windows。

◼ 统一事件源。Libevent 对 I/O 事件、信号和定时事件提供统一的处理。

◼ 线程安全。Libevent 使用 libevent_pthreads 库来提供线程安全支持。

基于 Reactor 模式的实现。

Libevent 使用模型

使用Libevent的基本流程:

根据一个简单的场景,了解其基本流程

1)首先初始化 libevent 库,并保存返回的指针

struct event_base * base = event_init();这个init底层其实调用的就是下面这个base_new

等于struct event_base *base=event_base_new();常用

// 创建event_base
struct event_base* event_base_new(void)

当我们创建一个event_base的时候,libevent会自动为我们选择最快的IO多路复用模型,Linux下一般会用epoll模型

实际上这一步相当于初始化一个 Reactor 实例;在初始化 libevent 后,就可以注册事件了。

2)初始化事件 event,设置回调函数和关注的事件

evtimer_set(&ev, timer_cb, NULL);

事实上这等价于调用 event_set(&ev, -1, 0, timer_cb, NULL);

event_set 的函数原型是: 

void event_set(struct event *ev, int fd, short event, void (*cb)(int, short, void *), void *arg) ev

ev:执行要初始化的 event 对象;

fd:该 event 绑定的“句柄”,对于信号事件,它就是关注的信号;

event:在该 fd 上关注的事件类型,它可以是 EV_READ, EV_WRITE, EV_SIGNAL;

cb:这是一个函数指针,当 fd 上的事件 event 发生时,调用该函数执行处理,它有三个参数, 调用时由 event_base 负责传入,按顺序,实际上就是 event_set 时的 fd, event 和 arg;

arg:传递给 cb 函数指针的参数; 由于定时事件不需要 fd,并且定时事件是根据添加时(event_add)的超时值设定的,因此 这里 event 也不需要设置。

这一步相当于初始化一个 event handler,在 libevent 中事件类型保存在 event 结构体中。 注意:libevent 并不会管理 event 事件集合,这需要应用程序自行管理;

3)设置 event 从属的 event_base

event_base_set(base, &ev);

这一步相当于指明 event 要注册到哪个 event_base 实例上;

上面的2 ,3其实都可以用下面的event_new,evtimer_new啊这些去初始化事件,总之set现在很少用了,set跟下面主要介绍的event_new一个道理,干的都是一个事情,参数都是一样的

4)是正式的添加事件的时候了

event_add(&ev, timeout);

基本信息都已设置完成,只要简单的调用 event_add()函数即可完成,其中 timeout 是定时值;

5)程序进入无限循环,等待就绪事件并执行事件处理

我们上面说到 event_base是一组event的集合,我们也可以将event事件注册到这个集合中。当需要事件监听的时候,我们就需要对这个event_base进行循环。

event_base_dispatch(base);//linux下一般就调用的io方法是epoll

返回值:0 表示成功退出  -1 表示存在错误信息。

6)上面event_new或者set了事件,下面就要释放

// 创建event_freevoid event_free(struct event *event); 

7)销毁event_base 

// 释放event_base_free
event_base_free(struct event_base* base);

 示例代码:

struct event ev; 
struct timeval tv; 
/*
struct timeval {long    tv_sec;         /* seconds */long    tv_usec;        /* and microseconds */
};
tv_sec 代表多少秒
tv_usec 代表多少微秒 1000000 微秒 = 1秒
*/void time_cb(int fd, short event, void *argc) 
{ printf("timer wakeup\n"); event_add(&ev, &tv); // reschedule timer 
} 
int main() 
{ struct event_base *base = event_init(); tv.tv_sec = 10; // 10s period tv.tv_usec = 0; evtimer_set(&ev, time_cb, NULL); event_add(&ev, &tv); event_base_dispatch(base); 
}

libevent 的核心,event 事件

event_base是事件的集合,负责事件的循环,以及集合的销毁。而event就是event_base中的基本单元:事件。

我们举一个简单的例子来理解事件。例如我们的socket来进行网络开发的时候,都会使用accept这个方法来阻塞监听是否有客户端socket连接上来,如果客户端连接上来,则会创建一个线程用于服务端与客户端进行数据的交互操作,而服务端会继续阻塞等待下一个客户端socket连接上来。客户端连接到服务端实际就是一种事件。

1. 创建一个事件event

struct event *event_new(struct event_base *base, evutil_socket_t fd,short events, event_callback_fn cb,void *arg);

参数:

  • base:即event_base
  • fd:文件描述符。
  • events:event关心的各种条件(事件类型)。
  • cb:回调函数。
  • arg:用户自定义的数据,可以传递到回调函数中去。

libevent是基于事件的,也就是说只有在事件到来的这种条件下才会触发当前的事件。例如:

  • fd文件描述符已准备好可写或者可读
  • fd马上就准备好可写和可读。
  • 超时的情况 timeout
  • 信号中断
  • 用户触发的事件
     

Libevent 支持的事件类型

 可以看出事件类型可以使用“|”运算符进行组合,需要说明的是,信号和I/O事件不能同时设置;

2. 释放event_free

真正的释放event的内存。 

void event_free(struct event *event);

event_del 清理event的内存。这个方法并不是真正意义上的释放内存。

当函数会将事件转为 非pending和非activing的状态。

int event_del(struct event *event);

3. 注册event

该方法将用于向event_base注册事件。

参数:ev 为事件指针;tv 为时间指针。当tv = NULL的时候则无超时时间。

函数返回:0表示成功 -1 表示失败。

int event_add(struct event *ev, const struct timeval *tv);

tv时间结构例子: 

struct timeval five_seconds = {5, 0};
event_add(ev1, &five_seconds);

或者像上面我最早示例的那种把结构体定义出来分别给他的成员赋值

struct timeval tv; 
/*
struct timeval {long    tv_sec;         /* seconds */long    tv_usec;        /* and microseconds */
};
tv_sec 代表多少秒
tv_usec 代表多少微秒 1000000 微秒 = 1秒
*/tv.tv_sec = 5; // 5s period tv.tv_usec = 0; event_add(ev1, &tv);

4. 信号事件 

信号事件也可以对信号进行事件的处理。用法和event_new类似。只不过处理的是信号而已。

// base --- event_base
// signum --- 信号,例如 SIGHUP
// callback --- 信号出现时调用的回调函数
// arg --- 用户自定义数据
evsignal_new(base, signum, cb, arg)//将信号 event 注册到 event_base
evsignal_add(ev, tv) // 清理信号 event
evsignal_del(ev) 

注意一些细节:

每一个事件event都需要通过event_new初始化生成。event_new生成的事件是在堆上分配的内存。
当一个事件通过event_add被注册到event_base上的时候,这个事件处于pending(等待状态),当只有有事件进来的时候,event才会被激活active状态,相关的回调函数就会被调用。
persistent 如果event_new中的what参数选择了EV_PERSIST,则是持久的类型持久的类型调用玩回调函数后,会继续转为pending状态,就会继续等待事件进来。大部分情况下会选择持久类型的事件。
而非持久的类型的事件,调用玩一次之后,就会变成初始化的状态。这个时候需要调用event_add 继续将事件注册到event_base上之后才能使用。
 

示例:设置定时器,从键盘获得SIGINT信号,打印其对应数字

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<event.h>
#include<assert.h>
#include<signal.h>
#include<sys/time.h>void sig_fun(int fd,short ev,void* arg)
{printf("sig=%d\n",fd);
}
void timeout(int fd,short ev,void *arg)
{printf("time out\n");
}int main()
{struct event_base *base=event_base_new();//创建实例assert(base!=NULL);//创建事件// struct event*sig_ev=evsignal_new(base,SIGINT,sig_fun,NULL);//定义信号事件struct event * sig_ev = event_new(base,SIGINT,EV_SIGNAL|EV_PERSIST,sig_fun,NULL);assert(sig_ev!=NULL);event_add(sig_ev,NULL);//添加事件到实例中//struct event* ev_time=evtimer_new(base,timeout,NULL);//定义定时器事件 struct event* ev_time = event_new(base,-1,EV_TIMEOUT|EV_PERSIST,timeout,NULL);struct timeval tv={5,0};event_add(ev_time,&tv);//添加事件到实例中//调用事件循环方法event_base_dispatch(base);//阻塞,底层调io方法event_free(ev_time);event_free(sig_ev);event_base_free(base);return 0;
}

Libevent 结构图

使用libevent库去实现tcp服务器

使用 libevent 库实现的 TCP 服务器代时,将监听 socket 和连接 socket 分别生成一个 Libevent 事件(指定其对应的回调函数),并将其添加到 Libevent 的一个 Base 中,执行事件 循环,检测事件发生。(客户端的代码与 select 部分客户端代码相同),代码示例如下:

#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<event.h>
#include<assert.h>
#include<signal.h>
#include<sys/time.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<netinet/in.h>int socket_init();
void recv_cb(int fd,short ev,void *arg)
{struct event**c_ev=(struct event**)arg;if(ev&EV_READ){char buff[128]={0};int n=recv(fd,buff,127,0);if(n<=0){//移除事件 event_delevent_free(*c_ev);free(c_ev);//关闭连接close(fd);printf("close one clinet\n");//释放事件占的空间 event_free->event_delreturn;}printf("recv(%d)=%s\n",fd,buff);send(fd,"ok",2,0);}
}
void accept_cb(int fd,short ev,void *arg)
{struct event_base* base=(struct event_base*)arg;if(ev&EV_READ){struct sockaddr_in caddr;int len=sizeof(caddr);int c=accept(fd,(struct sockaddr*)&caddr,&len);if(c<0){return ;}printf("accept c=%d\n客户端ip:(%s)已连接;客户端端口port:(%d)已连接\n",c,inet_ntoa(caddr.sin_addr),ntohs(caddr.sin_port)); struct event**c_ev=(struct event**)malloc(sizeof(struct event*));if(c_ev==NULL){return ;}*c_ev=event_new(base,c,EV_READ|EV_PERSIST,recv_cb,(void*)c_ev);if(*c_ev==NULL){return ;}event_add(*c_ev,NULL);}
}int main()
{int sockfd=socket_init();assert(sockfd!=-1);struct event_base* base=event_base_new();assert(base!=NULL);struct event* sock_ev=event_new(base,sockfd,EV_READ|EV_PERSIST,accept_cb,base);assert(sock_ev!=NULL);event_add(sock_ev,NULL);event_base_dispatch(base);//事件循环 阻塞event_free(sock_ev);event_base_free(base);
}int socket_init()
{int sockfd=socket(AF_INET,SOCK_STREAM,0);if(sockfd==-1){return sockfd;}struct sockaddr_in saddr;memset(&saddr,0,sizeof(saddr));saddr.sin_family=AF_INET;saddr.sin_port=htons(6000);saddr.sin_addr.s_addr=inet_addr("127.0.0.1");int res=bind(sockfd,(struct sockaddr*)&saddr,sizeof(saddr));if(res==-1){return -1;}res=listen(sockfd,5);if(res==-1){return -1;}return sockfd;
}


http://chatgpt.dhexx.cn/article/fASJRiTQ.shtml

相关文章

编译libevent

本文记录在win10编译libevent的过程 1.编译前准备zlib,openssl zlib网址 http://www.zlib.net/ 下载源码解压缩 打开vs的dos窗口 32位选择32位窗口这里选择64位,cd 到解压后的文件夹 执行 nmake /f win32/Makefile.msc 执行后可以看到目录下有lib文件和dll文件和测试文件…

libevent简要介绍

libevent库 开源。精简。跨平台&#xff08;Windows、Linux、Maxos、unix&#xff09;。专注于网络通信。 源码包安装&#xff1a; ./configure 检查安装环境 生成makefile make 生成.o和可执行文件 sudo make install …

Libevent库的简介

一、libevent是什么 Libevent 是一个用C语言编写的、轻量级的开源高性能事件通知库&#xff0c;主要有以下几个亮点&#xff1a;事件驱动&#xff08; event-driven&#xff09;&#xff0c;高性能;轻量级&#xff0c;专注于网络&#xff0c;不如 ACE 那么臃肿庞大&#xff1b…

libevent库学习(1)

一、初识 1、libevent介绍 Libevent 是一个用C语言编写的、轻量级的开源高性能事件通知库&#xff0c;主要有以下几个亮点&#xff1a;事件驱动&#xff08; event-driven&#xff09;&#xff0c;高性能;轻量级&#xff0c;专注于网络&#xff0c;不如 ACE 那么臃肿庞大&…

详解Libevent网络库

项目中要用到libevent&#xff0c;所以就自学了libevent&#xff0c;参考资料为张亮的《libevent源码深度剖析》和《linux高性能服务器编程》 Libevent简介 Libevent是开源社区一款高性能的I/O框架库&#xff0c;其具有如下特点&#xff1a; 1.跨平台支持。Libevent支持Linu…

Libevent库的介绍与应用

Libevent库 Libevent概述Libevent使用模型Libevent库使用示例Libevent事件类型和框架结构使用Libevent完成tcp服务端 Libevent概述 Libevent是开源社区的一款高性能的I/O框架库&#xff0c;使用Libevent的著名案例有&#xff1a;高性能的分布式内存对象缓存软件memcached,Goog…

Linux c 开发 - libevent

目录 一、event_base 1. 创建event_base 2. 查看IO模型 3. 销毁event_base 4. 事件循环 event loop 5. event_base的例子 二、event 事件 1. 创建一个事件event 2. 释放event_free 3. 注册event 4.event_assign 5. 信号事件 6. event细节 三、Socket实例 四、Bu…

安装svn客户端

下载地址 下载 TortoiseSVN 我的网盘地址&#xff1a; 百度网盘 请输入提取码 提取码&#xff1a; em9s 1、安装客户端 双击运行&#xff1a; 点击【next】 点击【next】 选择好路径后&#xff0c;点击【next】 点击【install】安装 点击【finish】完成安装。安装完成后重…

SVN客户端配置--客户端

一、使用客户端软件连接SVN服务器 在项目文件夹中单击右键–>TortoiseSVN–>Repo-browser,并在弹出框中贴入URL&#xff08;svn://服务器IP&#xff09; 在图中指定的位置单击右键选择checkput 点击OK点击OK SVN的三大指令 指令名称Checkout①连接到SVN服务器②更新服务…

SVN客户端(TortoiseSVN)安装及使用说明

&#x1f60a; 作者&#xff1a; 一恍过去 &#x1f496; 主页&#xff1a; https://blog.csdn.net/zhuocailing3390 &#x1f38a; 社区&#xff1a; Java技术栈交流 &#x1f389; 主题&#xff1a; SVN客户端(TortoiseSVN)安装及使用说明 ⏱️ 创作时间&#xff1a; 20…

ubuntu搭建SVN客户端,右击提交或更新

有时候我们会在Linux系统上搞开发&#xff0c;项目在SVN上&#xff0c;不搭建SVN环境搞起来比较麻烦&#xff0c;不用其他软件为载体实现右击提交更新等其他操作&#xff0c;不使用VsCode,IDE全家桶等为载体搭建插件形式的SVN&#xff0c;完全独立搭建 1&#xff1a;安装SVN客…

SVN客户端下载及安装

SVN客户端地址&#xff1a;https://osdn.net/projects/tortoisesvn/# 方法/步骤 官网下载tortoisesvn&#xff0c;下载完成之后&#xff0c;如下图所示&#xff1a; 点击安装进入安装界面&#xff0c;点击接受协议&#xff0c;如下图所示&#xff1a; 选择安装的地址&…

【转】Ubuntu中SVN客户端安装+使用

转自&#xff1a;Ubuntu中SVN客户端安装使用_三少GG-CSDN博客 1、 安装 svn客户端&#xff1a; apt-get install subversion&#xff0c;然后根据提示一步一步&#xff0c;就完成了 svn的安装。当然&#xff0c;也可以源码安装 svn&#xff0c;下载 subversion 一个最新版本的源…

SVN客户端

SVN是Subversion的简称&#xff0c;是一个开源的代码版本控制系统 SVN客户端操作 checkout 检出&#xff1a;初次下载-第一次连接svn服务器时候需要下载对应仓库的数据&#xff08;如果仓库中有数据的话&#xff09;add 新增&#xff1a;新增数据到svn服务器update 更新&#x…

【SVN】windows下的SVN客户端访问ubuntu下的SVN服务器

目录 第一部分 windows创建本地版本库、连接ubuntu的SVN服务器 步骤0&#xff1a; 步骤一&#xff1a;创建windows本地版本库 步骤二&#xff1a;checkout检测 步骤三&#xff1a;输入之前配置的用户名和密码 第二部分 windows上传文件至SVN服务器 步骤一&#xff1a;添加…

SVN 客户端下载与安装

SVN 客户端下载与安装 svn 客户端下载地址&#xff1a;https://tortoisesvn.net/downloads.zh.html 1. 网址截图 2. 汉化包下载安装 下载完毕&#xff0c;双击安装&#xff0c;下一步即可&#xff08;可根据需要进行安装路径选择&#xff09;

windows环境下的svn客户端(VisualSVN Server),服务端(TortoiseSVN),中文化(TortoiseSVN LanguagePack_1.9.7.2)安装和使用(超详细)

一、下载安装包 1.VisualSVN server 服务端下载&#xff1a;VisualSVN Server 2.svn客户端下载&#xff1a;TortoiseSVN 3.中文化包下载&#xff1a;TortoiseSVN LanguagePack_1.9.7.27907-x64-zh_CN中文包 二、安装软件 1.安装VisualSVN server 服务端&#xff1a; &#xf…

SVN客户端安装及操作文档

第一部分&#xff1a;客户端安装 1 在D盘创建svnclient文件夹&#xff0c;并将svn安装包复制到svnclient文件夹中 2 客户端安装 双击文件“TortoiseSVN-1.14.1.29085-x64-svn-1.14.1.msi” 点击“Next” 点击“Next” 点击“Browse”选择安装路径&#xff0c;这里将安装路径…

12306图片验证码效果实现

效果如下&#xff1a; 原理&#xff1a;从服务器获取的图片上&#xff0c;我们画上自己的图片&#xff0c;然后再显示出来。 核心代码如下在处理imageview点击事件前做如下操作&#xff1a; Bitmap bitmap BitmapFactory.decodeResource(getResources(), R.mipmap.yanzheng)…

利用神经网络识别12306验证码—(六)模型应用以及12306实战测试

模型训练好之后&#xff0c;就可以应用于新图片的预测了。比如现在有下面这么一张12306的验证码&#xff0c;预测工作也分为两部分&#xff1a;上半部分的文字预测、下半部分的图片预测&#xff0c;将两部分划分开之后分别加载各自的模型进行预测。 文字预测&#xff1a;需要把…