(转)libevent介绍及示例

一、Libevent简介

libevent是一个基于事件触发的网络库,适用于windows、linux、bsd等多种平台,内部使用select、epoll、kqueue等系统调用管理事件机制。官网:http://libevent.org/

特点:

事件驱动,高性能;

轻量级,专注于网络,不如ACE那么臃肿庞大,只提供了简单的网络API的封装,线程池,内存池,递归锁等均需要自己实现;

开放源码,代码相当精炼、易读;

跨平台,支持Windows、Linux、BSD和Mac OS;

支持多种I/O多路复用技术(epoll、poll、dev/poll、select和kqueue等),在不同的操作系统下,做了多路复用模型的抽象,可以选择使用不同的模型,通过事件函数提供服务;

支持I/O,定时器和信号等事件;

采用Reactor模式;

二、源码组织结构

Libevent 的源代码虽然都在一层文件夹下面,但是其代码分类还是相当清晰的,主要可分为头文件、内部使用的头文件、辅助功能函数、日志、libevent框架、对系 统I/O多路复用机制的封装、信号管理、定时事件管理、缓冲区管理、基本数据结构和基于libevent的两个实用库等几个部分,有些部分可能就是一个源文件。

1)头文件

主要就是event.h:事件宏定义、接口函数声明,主要结构体event的声明;

2)内部头文件

xxx-internal.h:内部数据结构和函数,对外不可见,以达到信息隐藏的目的;

3)libevent框架

event.c:event整体框架的代码实现;

4)对系统I/O多路复用机制的封装

epoll.c:对epoll的封装;

select.c:对select的封装;

devpoll.c:对dev/poll的封装;

kqueue.c:对kqueue的封装;

5)定时事件管理

min-heap.h:其实就是一个以时间作为key的小根堆结构;

6)信号管理

signal.c:对信号事件的处理;

7)辅助功能函数

evutil.h 和evutil.c:一些辅助功能函数,包括创建socket pair和一些时间操作函数:加、减和比较等。

8)日志

log.h和log.c:log日志函数

9)缓冲区管理

evbuffer.c和buffer.c:libevent对缓冲区的封装;

10)基本数据结构

compat/sys下的两个源文件:queue.h是libevent基本数据结构的实现,包括链表,双向链表,队列等;_libevent_time.h:一些用于时间操作的结构体定义、函数和宏定义;

11)实用网络库

http和evdns:是基于libevent实现的http服务器和异步dns查询库;

三、示例

1、获取版本

// gcc getVersion.c -o getVersion -levent
#include <event.h>
#include <stdio.h>
int main()
{
const char *version = event_get_version();
printf("%s\n",version);
return 0;
}

2、timer程序

// gcc timer.c -o timer -levent
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <event2/event.h>
#include <event2/event_struct.h>
#define N 300
#define BUFLEN 256
struct timeval lasttime;
struct ST_EventWithDescription
{
struct event *p_event;
int time_interval;
char lable[BUFLEN];
};
static void timeout_cb(evutil_socket_t fd, short event, void *arg)
{
struct timeval newtime, difference;
struct ST_EventWithDescription *pSTEvent = arg;
struct event *timeout = pSTEvent->p_event;
double elapsed;
evutil_gettimeofday(&newtime, NULL);
evutil_timersub(&newtime, &lasttime, &difference);
elapsed = difference.tv_sec + (difference.tv_usec / 1.0e6);
printf("%s called at %d: %.3f seconds since my last work.\n",
(char*)pSTEvent->lable,(int)newtime.tv_sec, elapsed);
lasttime = newtime;
struct timeval tv;
evutil_timerclear(&tv);
tv.tv_sec = pSTEvent->time_interval;
event_add(timeout, &tv);
}
void setParam(struct ST_EventWithDescription *stEventDescription,
struct event *m_event,int time_interval,char* m_lable)
{
stEventDescription->p_event = m_event;
stEventDescription->time_interval = time_interval;
memset(stEventDescription->lable,0,sizeof(stEventDescription->lable));
memcpy(stEventDescription->lable,m_lable,strlen(m_lable)+1);
}
void setTimeIntervalArr(int *arr,int n)
{
int i;
srand(time(NULL));
for(i=0; i<n; ++i)
{
*(arr+i) = rand()%n + 1;
//*(arr+i) = i+1;
}
}
int main(int argc, char **argv)
{
struct event timeout[N];
struct ST_EventWithDescription stEvent[N];
int time_interval[N];
int i=0;
struct timeval tv;
struct event_base *base;
int flags = 0;
setTimeIntervalArr(time_interval,N);
base = event_base_new();
evutil_timerclear(&tv);
for(i=0; i<N; ++i)
{
char buf[BUFLEN]= {0};
sprintf(buf,"task%d",i+1);
setParam(stEvent+i,timeout+i,time_interval[i],buf);
event_assign(timeout+i, base, -1, flags, timeout_cb, (void*)(stEvent+i));
event_add(timeout+i, &tv);
}

evutil_gettimeofday(&lasttime, NULL);
event_base_dispatch(base);
return (0);
}

3、socket程序

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <netinet/in.h>
#include <netinet/tcp.h>
#include <event.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <errno.h>
#include <fcntl.h>
static short ListenPort = 8080;
static long ListenAddr = INADDR_ANY;//任意地址的值就是0
static int MaxConnections = 1024;
static int ServerSocket;
static struct event ServerEvent;//创建event
//不论在什么平台编写网络程序,都应该使用NONBLOCK将一个socket设置成非阻塞模式。这样可以保证你的程序至少不会在recv/send/accept/connect这些操作上发生block从而将整个网络服务都停下来
int SetNonblock(int fd)
{
int flags;

if ((flags = fcntl(fd, F_GETFL)) == -1) { //用来操作文件描述符的一些特性
return -1;
}
if (fcntl(fd, F_SETFL, flags | O_NONBLOCK) == -1) {
return -1;
}
return 0;
}
//这个函数当客户端的socket可读时由libevent调用
void ServerRead(int fd, short ev, void *arg)
{
struct client *client = (struct client *)arg;
u_char buf[8196];
int len, wlen;
//会把参数fd 所指的文件传送count个字节到buf指针所指的内存中
len = read(fd, buf, sizeof(buf));
if (len == 0) {
/* 客户端断开连接,在这里移除读事件并且释放客户数据结构 */
printf("disconnected\n");
close(fd);
event_del(&ServerEvent);
free(client);
return;
} else if (len < 0) {
/* 出现了其它的错误,在这里关闭socket,移除事件并且释放客户数据结构 */
printf("socket fail %s\n", strerror(errno));
close(fd);
event_del(&ServerEvent);
free(client);
return;
}
/* 
为了简便,我们直接将数据写回到客户端。通常我们不能在非阻塞的应用程序中这么做,
我们应该将数据放到队列中,等待可写事件的时候再写回客户端。 
如果使用多个终端进行socket连接会出现错误socket fail Bad file descriptor
*/
wlen = write(fd, buf, len);
if (wlen < len) {
printf("not all data write back to client\n");
}
return;
}
/*
当有一个连接请求准备被接受时,这个函数将被libevent调用并传递给三个变量: 
int fd:触发事件的文件描述符. 
short event:触发事件的类型EV_TIMEOUT,EV_SIGNAL, EV_READ, or EV_WRITE. 
void* :由arg参数指定的变量. 
*/
void ServerAccept(int fd, short ev, void *arg)
{
int cfd;
struct sockaddr_in addr;
socklen_t addrlen = sizeof(addr);
int yes = 1;
int retval;
//将从连接请求队列中获得连接信息,创建新的套接字,并返回该套接字的文件描述符。
//新创建的套接字用于服务器与客户机的通信,而原来的套接字仍然处于监听状态。
//该函数的第一个参数指定处于监听状态的流套接字
cfd = accept(fd, (struct sockaddr *)&addr, &addrlen);
if (cfd == -1) {
printf("accept(): can not accept client connection");
return;
}
if (SetNonblock(cfd) == -1) {
close(cfd);
return;
}
//设置与某个套接字关联的选项
//参数二 IPPROTO_TCP:TCP选项
//参数三 TCP_NODELAY 不使用Nagle算法 选择立即发送数据而不是等待产生更多的数据然后再一次发送
// 更多参数TCP_NODELAY 和 TCP_CORK
//参数四 新选项TCP_NODELAY的值
if (setsockopt(cfd, IPPROTO_TCP, TCP_NODELAY, &yes, sizeof(yes)) == -1) {
printf("setsockopt(): TCP_NODELAY %s\n", strerror(errno));
close(cfd);
return;
}
event_set(&ServerEvent, cfd, EV_READ | EV_PERSIST, ServerRead, NULL);
event_add(&ServerEvent, NULL);

printf("Accepted connection from %s\n", inet_ntoa(addr.sin_addr));
}
int NewSocket(void)
{
struct sockaddr_in sa;
//socket函数来创建一个能够进行网络通信的套接字。
//第一个参数指定应用程序使用的通信协议的协议族,对于TCP/IP协议族,该参数置AF_INET;
//第二个参数指定要创建的套接字类型
//流套接字类型为SOCK_STREAM、数据报套接字类型为SOCK_DGRAM、原始套接字SOCK_RAW
//第三个参数指定应用程序所使用的通信协议。
ServerSocket = socket(AF_INET, SOCK_STREAM, 0);
if (ServerSocket == -1) {
printf("socket(): can not create server socket\n");
return -1;
}
if (SetNonblock(ServerSocket) == -1) {
return -1;
}
//清空内存数据
memset(&sa, 0, sizeof(sa));
sa.sin_family = AF_INET;
//htons将一个无符号短整型数值转换为网络字节序
sa.sin_port = htons(ListenPort);
//htonl将主机的无符号长整形数转换成网络字节顺序
sa.sin_addr.s_addr = htonl(ListenAddr);
//(struct sockaddr*)&sa将sa强制转换为sockaddr类型的指针
/*struct sockaddr 
数据结构用做bind、connect、recvfrom、sendto等函数的参数,指明地址信息。
但一般编程中并不直接针对此数据结构操作,而是使用另一个与sockaddr等价的数据结构 struct sockaddr_in
sockaddr_in和sockaddr是并列的结构,指向sockaddr_in的结构体的指针也可以指向
sockadd的结构体,并代替它。也就是说,你可以使用sockaddr_in建立你所需要的信息,
在最后用进行类型转换就可以了
*/
//bind函数用于将套接字绑定到一个已知的地址上
if (bind(ServerSocket, (struct sockaddr*)&sa, sizeof(sa)) == -1) {
close(ServerSocket);
printf("bind(): can not bind server socket");
return -1;
}

//执行listen 之后套接字进入被动模式
//MaxConnections 连接请求队列的最大长度,队列满了以后,将拒绝新的连接请求
if (listen(ServerSocket, MaxConnections) == -1) {
printf("listen(): can not listen server socket");
close(ServerSocket);
return -1;

时间: 2024-12-11 05:39:04

(转)libevent介绍及示例的相关文章

[libevent]Libevent介绍与编译

libevent介绍 libevent是一个轻量级的,开源高性能的,基于事件触发的网络库,适用于windows.linux.bsd等多种平台,内部使用select.epoll.kqueue等系统调用管理事件机制. 编译库代码,编译脚本会判断OS支持哪种类型的事件机制(select.epoll或kqueue),然后条件编译相应代码,供上层使用的接口仍然是保持统一的. 有许多开源项目使用libevent,例如memcached.使用libevent,使得memcached可以适应多种操作系统.Lib

SharePoint 2013 Word 转换PDF服务介绍及示例

原文:SharePoint 2013 Word 转换PDF服务介绍及示例 前言:在SharePoint使用过程中,经常会发现将文档进行格式转换的需求,之前,看到SharePoint 2013有将PPT转换PDF文档的服务,后来,才发现SharePoint 2010开始,就有将Word文档转换PDF的服务了,用起来还是不错的. 下面,我们就来看看这个服务,如何将一个Word文档库批量转换成PDF文件,或者,将单一Word文档转换为PDF文档. 一.             效果展示 新建测试Wor

Linux下libevent安装与示例

用wget指令直接下载libevent:# wget http://www.monkey.org/~provos/libevent-1.2.tar.gz 然后,在配置时需要指定一个安装路径,即./configure –prefix=/usr:然后make:然后make install: 详细的方法如下: # cd /tmp # wget http://www.monkey.org/~provos/libevent-1.2.tar.gz # tar zxvf libevent-1.2.tar.gz

smarty 基本介绍及示例

什么是smarty? Smarty是一个使用PHP写出来的模板引擎,是业界最著名的PHP模板引擎之一.Smarty分离了逻辑代码和外在的内容,提供一种易于管理和使用的方法,用来将原本与HTML代码混杂在一起PHP代码逻辑分离.Smarty工作的目的是要使PHP程序员同前端人员分离,使程序员改变程序的逻辑内容不会影响到前端人员的页面设计,前端人员重新修改页面不会影响到程序的程序逻辑,这在多人合作的项目中显的尤为重要. 为什么会诞生smarty? 例如在一个公司,一个应用程序的开发流程如下:在提交计

Erlang基础 -- 介绍 -- Wordcount示例演示

在前两个blog中,已经说了Erlang的历史.应用场景.特点,这次主要演示一个Wordcount的示例,就是给定一个文本文件,统计这个文本文件中的单词以及该单词出现的次数. 今天和群友们讨论了一个问题,突然一下子就上升到哲学角度上了,装逼装大发了. PS:图片中有错别字,%s/财务和其他9个月/财务和其他9个人/g 不过真心想说的一点是,把Erlang系统,映射到现实中,很多奇葩问题,就能迎刃而解了.所以,在下面的简要设计中,我就尽可能的代入一下现实世界吧. 环境安装 mac 的话,用brew

JAVA HashMap详细介绍和示例

第1部分 HashMap介绍 HashMap简介 HashMap 是一个散列表,它存储的内容是键值对(key-value)映射. HashMap 继承于AbstractMap,实现了Map.Cloneable.java.io.Serializable接口. HashMap 的实现不是同步的,这意味着它不是线程安全的.它的key.value都可以为null.此外,HashMap中的映射不是有序的. HashMap 的实例有两个参数影响其性能:"初始容量" 和 "加载因子&quo

unity Dotween插件的简单介绍及示例代码

unity里面做插值动画的插件有许多,比较常见的有itween.hotween.dotween.根据大家的反馈和实际体验来说,dotween插件在灵活性.稳定性.易用性上都十分突出.这里简单介绍下它的用法,并在后文做了一些效果示例,还是不错的. 所谓"插值动画",顾名思义就是在两个值中插入其他的值来实现动画.原理非常简单,比如想让某个物体从A地到达B地,我们只知道A和B的坐标,插值动画就可以根据"缓动函数"确定A.B间的其他值,来实现物体从A到B的"运动过

ArcSDE SDK For Java二次开发介绍、示例

在一个工作中,遇到了需要java后台来查询ArcGIS 中用到的Oracle数据库空间数据,由于对ArcGIS空间数据首次接触,只知道Oracle可以使用ST_GEOMETRY字段存储,如下图 但是查询时会发现这个ST_GEOMETRY字段会在结果中出现个多个子的字段,对于arcgis地理知识了解甚少,不知道单独查询一个坐标怎么弄了,有些朋友说需要配置监听文件. 不管怎么说,至少参考了一个大牛的文章,http://blog.csdn.net/linghe301/article/details/8

C语言 运算符详细介绍及示例代码

C 运算符 运算符是一种告诉编译器执行特定的数学或逻辑操作的符号.C 语言内置了丰富的运算符,并提供了以下类型的运算符: 算术运算符 关系运算符 逻辑运算符 位运算符 赋值运算符 杂项运算符 本章将逐一介绍算术运算符.关系运算符.逻辑运算符.位运算符.赋值运算符和其他运算符. 算术运算符 下表显示了 C 语言支持的所有算术运算符.假设变量 A 的值为 10,变量 B 的值为 20,则: 运算符 描述 实例 + 把两个操作数相加 A + B 将得到 30 - 从第一个操作数中减去第二个操作数 A