Libevent库的学习

news/2024/5/20 4:15:19/文章来源:https://blog.csdn.net/weixin_51609435/article/details/127532516

目录

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();

当我们创建一个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 实例上;

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

event_add(&ev, timeout);

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

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

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

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

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

示例代码:

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) 

5. 销毁event_base 

void event_base_free(struct event_base *base);

注意一些细节:

每一个事件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_init();//创建实例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_init();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://www.luyixian.cn/news_show_218637.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【目标检测】【边界框回归】Bounding-Box regression

最近开始看目标检测的论文&#xff0c;第一篇为R-CNN论文&#xff0c;是两阶段目标检测的开山奠基之作。论文中的损失函数包含了边界框回归&#xff0c;且在R-CNN论文里面有详细的介绍。 一、为什么要做边界框回归&#xff1f; 对于上图&#xff0c;绿色的框表示Ground Truth&…

【VIO】第2讲 基于优化的IMU

第2讲 基于优化的 IMU 与视觉信息融合 1.最小二乘问题求解 &#xff08;1&#xff09;最小二乘基础概念 ​ 1 定义&#xff1a;找到一个n维的变量 x∈Rnx \in R^nx∈Rn &#xff0c;使得损失函数 F(x)F(x)F(x) 取得局部最小值&#xff1a; F(x)12∑i1m(fi(x))2F(x) \frac{1…

Word控件Spire.Doc 【文本】教程(5) ;从 Word 文档中的文本框中提取文本

文本框的目的是允许用户输入程序要使用的文本信息。也可以从文本框中提取现有的文本信息。以下指南重点介绍如何通过Spire.Doc for .NET从 C# 中 Word 文档的文本框中提取文本。 Spire.Doc for.NET 最新下载&#xff08;qun:767755948&#xff09;https://www.evget.com/produ…

3、Java对象相关

目录JVM内存分配机制对象的创建对象大小与指针压缩java对象的指针压缩指针压缩的原因分代回收机制分代GC分类对象内存分配栈上分配逃逸分析标量替换标量与聚合量Eden区分配大对象分配老年代分配对象动态年龄判断老年代空间分配担保机制对象的内存布局对象的访问定位对象内存回收…

WebDAV之葫芦儿·派盘+一刻日记

一刻日记 支持webdav方式连接葫芦儿派盘。 是一款强大的记录软件,通过平台可以随意的记录重要的事情,让用户在平台里能获得更多的帮助,实时的解决你的记录需求,让你可以更好的进行使用;在使用的过程中,用户可以记录当天重要的事情,把你的感想更好的记录在平台里,让用…

js-键盘事件

onkeydown:按键被按下 onkeyup:按键被松开 事件绑定的对象&#xff1a;键盘事件一般绑定给可以获取焦点的对象或者document对象 焦点&#xff1a;光标在闪的&#xff1a;比如input标签 如果一直按按键不松手&#xff0c;按键会一直被触发 当&#xff1a;onkeydown连续触发时…

后端php项目和数据库启动

有两种方法可以启动 1.使用小皮面板 ①启动php项目开启后端网站 可去官网下载 下载后就能使用了 官网地址&#xff1a;小皮面板(phpstudy) - 让天下没有难配的服务器环境&#xff01; 下载完成后打开 php项目需要启动apache 创建一个php项目的网站 注意这里要写public 点击…

亚马逊云 RDB数据库故障转移(多可用区)

RDB关系数据库(Relational Database,RDB) 创建名为VPC for RDS的vpc 两个可用区,两组公内网创建安全组创建RDS数据库实例用的数据库子网组创建RDS数据库实例创建数据库连接RDS数据库实例并给数据库test添加数据 1.创建安全组2.创建用来连接数据库实例的EC2选择vpc for rds那…

MyBatis 环境搭建配置全过程【IDEA】

文章目录一、MyBatis 介绍二、MyBatis 环境搭建1.MyBatis 下载2.配置 jdk 版本3.创建 Maven 工程4.IDEA 连接数据库5.项目文件构架6.引入相关依赖7.命令行创建数据库8.数据库配置文件9.核心配置文件三、入门测试程序1.创建表准备数据2.创建 POJO 实体3.创建映射文件4.修改核心配…

將一個react+nodejs聊天軟件前後端項目進行docker打包並運行

文章目录1概述2将react前端打包入docker2.1打包react项目2.2nginx配置2.3创建Docker镜像2.4打包和运行2.5上传dockerhub3将nodejs打包入dockerDockerfile文件.dockerignore 文件打包和运行上传dockerhub1概述 https://gitee.com/chuge325/practise–chat-app-react-nodejs.git…

爱上源码,重学Spring IoC深入

回答&#xff1a; 我们为什么要学习源码&#xff1f; 1、知其然知其所以然 2、站在巨人的肩膀上&#xff0c;提高自己的编码水平 3、应付面试1.1 Spring源码阅读小技巧 1、类层次藏得太深&#xff0c;不要一个类一个类的去看&#xff0c;遇到方法该进就大胆的进 2、更不要一行…

左程云老师算法课笔记( 四)

前言 仅记录学习笔记&#xff0c;如有错误欢迎指正。 啊啊&#xff0c;才发现二被我挤掉了&#xff0c;有空补下&#xff01; 一、图&#xff1a; 图的深度优先遍历&#xff1a;&#xff08;和二叉树的区别就是有环&#xff0c;不能重复打印&#xff09;&#xff08;Queue队…

网课搜题接口-查题校园题库系统

网课搜题接口-查题校园题库系统 本平台优点&#xff1a; 多题库查题、独立后台、响应速度快、全网平台可查、功能最全&#xff01; 1.想要给自己的公众号获得查题接口&#xff0c;只需要两步&#xff01; 2.题库&#xff1a; 查题校园题库&#xff1a;查题校园题库后台&…

全球名校AI课程库(28)| MIT麻省理工 · 基因组学机器学习课程『Machine Learning for Genomics』

&#x1f3c6; 课程学习中心 | &#x1f6a7; AI生物医疗课程合辑 | &#x1f30d; 课程主页 | &#x1f4fa; 中英字幕视频 | &#x1f680; 项目代码解析 课程介绍 MIT 6.047/6.878是全球顶校麻省理工开设的基因组学与机器学习的交叉专业课程。课程以基因组学为主要应用领域…

智慧城市万亿级蓝海赛道机遇何在?

工商业的发展&#xff0c;为人类居住历史增添了“城市”这一全新的选项。从春秋战国时期的“货市”&#xff0c;到13世纪地中海沿岸星罗棋布的都市&#xff0c;风格迥异的城市为身处不同时代的居民提供了栖居之地。仅在中国&#xff0c;城市就以不到6%的土地面积&#xff0c;维…

个人征信预测

个人征信预测 --数据分析项目报一、项目概述 通过脱敏的现有数据&#xff0c;如&#xff1a;用户基本身份信息&#xff0c;消费行为&#xff0c;银行还款等&#xff0c;进行数据处理特征&#xff0c;选取并建立逾期预测模型&#xff0c;预测用户是否会逾期还款。二、项目概述数…

SSD目标检测网络ONNX推理,为tensorrt推理做准备【附代码】

本篇文章是实现SSD的onnx推理&#xff0c;主要是为后期tensorrt推理打下基础&#xff0c;YOLOv4以及YOLOv5的tensorrt推理可以看我之前的文章。 SSD的代码我这里下载的是b站up主Bubbliiiing的pytorch版SSD&#xff0c;大家可自行下载【我这里就不传代码了&#xff0c;等最近把…

期货开户用心服务每个客户

用心服务每一个客户&#xff01;以信为本&#xff0c;点石成金&#xff01; 蓄之既久&#xff0c;其发必速 如果价格连续多天在—个狭窄的幅度内升降&#xff0c;在图表上形成一幅有如建筑地盘布满地基桩的图景&#xff0c;习惯上称之为密集区&#xff0c;亦即专家所说的技术…

【GraphQL】Node + Postgres + adminer实现demo应用

1、程序目录 在第一级目录下存在三个文件&#xff0c; db.sql用于创建tables和demo数据&#xff0c;可以直接在adminer里登录执行sql语句进行创建&#xff0c;可以看到如下图绿色部分的执行结果 docker-compose.yaml用于为node、postgres和adminer分别创建一个容器&#xff0…

数明SLM27517能驱动MOSFET和IGBT功率开关 低侧栅极驱动器兼容UCC27517

SLM27517 单通道&#xff0c;高速&#xff0c;低侧栅极驱动器器件可以有效地驱动MOSFET和IGBT功率开关。使用设计其固有地最小化击穿电流&#xff0c;可以源汇高峰值电流脉冲转换为电容性负载轨对轨驱动能力非常小传播延迟通常为15ns。可提供4 A电源&#xff0c;5 A接收器12 V …