著作权归作者所有。
商业转载请联系作者获得授权,非商业转载请注明出处。
作者:auxten
链接:http://zhuanlan.zhihu.com/auxten/20315482
来源:知乎 /*
这是一个示例性质的libevent的程序,监听在TCP的9995端口。
当连接建立成功后,它将会给Client回应一个消息"Hello, World!\n"
发送完毕后就将连接关闭。 程序也处理了SIGINT (ctrl-c)信号,收到这个信号后优雅退出程序。 这个程序也用到了一些libevent比较高级的API:“bufferevent”
这套API将buffer的“水位线”也抽象成了event来处理,灵感应该是来自
Windows平台的IOCP。
*/ // 引入常用Linux系统头文件
#include <string.h>
#include <errno.h>
#include <stdio.h>
#include <signal.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/socket.h> // 引入libevent 2.x相关的头文件
#include <event2/bufferevent.h>
#include <event2/buffer.h>
#include <event2/listener.h>
#include <event2/util.h>
#include <event2/event.h> // 定义字符串常量,将会回应给Client用
static const char MESSAGE[] = "Hello, World!\n"; // server监听的端口
static const int PORT = ; // 定义几个event callback的prototype(原型)
static void listener_cb(struct evconnlistener * , evutil_socket_t,
struct sockaddr * , int socklen, void * );
static void conn_writecb(struct bufferevent * , void * );
static void conn_eventcb(struct bufferevent * , short, void * );
static void signal_cb(evutil_socket_t, short, void * ); // 定义标准的main函数
int
main(int argc, char ** argv)
{
// event_base是整个event循环必要的结构体
struct event_base * base;
// libevent的高级API专为监听的FD使用
struct evconnlistener * listener;
// 信号处理event指针
struct event * signal_event;
// 保存监听地址和端口的结构体
struct sockaddr_in sin; // 分配并初始化event_base
base = event_base_new();
if (!base) {
// 如果发生任何错误,向stderr(标准错误输出)打一条日志,退出
// 在C语言里,很多返回指针的API都以返回null为出错的返回值
// if (!base) 等价于 if (base == null)
fprintf(stderr, "Could not initialize libevent!\n");
return ;
} // 初始化sockaddr_in结构体,监听在0.0.0.0:9995
memset(&sin, , sizeof(sin));
sin.sin_family = AF_INET;
sin.sin_port = htons(PORT); // bind在上面制定的IP和端口,同时初始化listen的事件循环和callback:listener_cb
// 并把listener的事件循环注册在event_base:base上
listener = evconnlistener_new_bind(base, listener_cb, (void * )base,
LEV_OPT_REUSEABLE|LEV_OPT_CLOSE_ON_FREE, -,
(struct sockaddr*)&sin,
sizeof(sin)); if (!listener) {
// 如果发生任何错误,向stderr(标准错误输出)打一条日志,退出
fprintf(stderr, "Could not create a listener!\n");
return ;
} // 初始化信号处理event
signal_event = evsignal_new(base, SIGINT, signal_cb, (void * )base); // 把这个callback放入base中
if (!signal_event || event_add(signal_event, NULL)<) {
fprintf(stderr, "Could not create/add a signal event!\n");
return ;
} // 程序将在下面这一行内启动event循环,只有在调用event_base_loopexit后
// 才会从下面这个函数返回,并向下执行各种清理函数,导致整个程序退出
event_base_dispatch(base); // 各种清理free
evconnlistener_free(listener);
event_free(signal_event);
event_base_free(base); printf("done\n");
return ;
} // 监听端口的event callback
static void
listener_cb(struct evconnlistener * listener, evutil_socket_t fd,
struct sockaddr * sa, int socklen, void * user_data)
{
struct event_base * base = user_data;
struct bufferevent * bev; // 新建一个bufferevent,设定BEV_OPT_CLOSE_ON_FREE,
// 保证bufferevent被free的时候fd也会被关闭
bev = bufferevent_socket_new(base, fd, BEV_OPT_CLOSE_ON_FREE);
if (!bev) {
fprintf(stderr, "Error constructing bufferevent!");
event_base_loopbreak(base);
return;
}
// 设定写buffer的event和其它event
bufferevent_setcb(bev, NULL, conn_writecb, conn_eventcb, NULL);
// 开启向fd中写的event
bufferevent_enable(bev, EV_WRITE);
// 关闭从fd中读写入buffer的event
bufferevent_disable(bev, EV_READ);
// 向buffer中写入"Hello, World!\n"
// 上面的操作保证在fd可写时,将buffer中的内容写出去
bufferevent_write(bev, MESSAGE, strlen(MESSAGE));
} // 每次fd可写,数据非阻塞写入后,会雕也难怪conn_writecb
// 这个函数每次检查eventbuffer的剩余大小,如果为0
// 表示数据已经全部写完,将eventbuffer free掉
// 由于在上面设定了BEV_OPT_CLOSE_ON_FREE,所以fd也会被关闭
static void
conn_writecb(struct bufferevent * bev, void * user_data)
{
struct evbuffer * output = bufferevent_get_output(bev);
if (evbuffer_get_length(output) == ) {
printf("flushed answer\n");
bufferevent_free(bev);
}
} // 处理读、写event之外的event的callback
static void
conn_eventcb(struct bufferevent * bev, short events, void * user_data)
{
if (events & BEV_EVENT_EOF) {
// Client端关闭连接
printf("Connection closed.\n");
} else if (events & BEV_EVENT_ERROR) {
// 连接出错
printf("Got an error on the connection: %s\n",
strerror(errno));
}
// 如果还有其它的event没有处理,那就关闭这个bufferevent
bufferevent_free(bev);
} // 信号处理event,收到SIGINT (ctrl-c)信号后,延迟2s退出event循环
static void
signal_cb(evutil_socket_t sig, short events, void * user_data)
{
struct event_base * base = user_data;
struct timeval delay = { , }; printf("Caught an interrupt signal; exiting cleanly in two seconds.\n"); event_base_loopexit(base, &delay);
}

示例的libevent的程序的更多相关文章

  1. 分布式 PostgreSQL 集群(Citus)官方示例 - 多租户应用程序实战

    如果您正在构建软件即服务 (SaaS) 应用程序,您可能已经在数据模型中内置了租赁的概念. 通常,大多数信息与租户/客户/帐户相关,并且数据库表捕获这种自然关系. 对于 SaaS 应用程序,每个租户的 ...

  2. 课程作业01:模仿JavaAppArguments.java示例,编写一个程序,此程序从命令行接收多个数字,求和之后输出结果。

    1.设计思想: 首先是从JavaAppArguments.java示例开始,此示例已打印参数,定义数字 之和和作为存储单位的整型,然后将输入参数的字符串转化为整型,之后求和即可. 2.程序流程图: 3 ...

  3. 模仿JavaAppArguments.java示例,编写一个程序,此程序从命令行接收多个数 字,求和之后输出结果,写出其的设计思想、程序流程图、源程序代码。

    一 设计思想 首先现在file中建立一个类,并把任务名和类名写上(注意类名的大写):第二步则是参数的输入,并且定义求和变量:第三步则是对参数数据类型的要求,要把字符类型转化为整数类型并输出(这也是本道 ...

  4. 用inno Setup做应用程序安装包的示例脚本(.iss文件)(

    用innoSetup做应用程序安装包的示例脚本(.iss文件),具体要看innoSetup附带的文档,好象是pascal语言写的脚本. 示例1(应用程序.exe,客户端安装): ;{089D6802- ...

  5. 使用 libevent 和 libev 提高网络应用性能

    使用 libevent 和 libev 提高网络应用性能 Martin C. Brown, 作家, Freelance 简介: 构建现代的服务器应用程序需要以某种方法同时接收数百.数千甚至数万个事件, ...

  6. libevent 和 libev 提高网络应用性能

    构建现代的服务器应用程序需要以某种方法同时接收数百.数千甚至数万个事件,无论它们是内部请求还是网络连接,都要有效地处理它们的操作.有许多解决方 案,但是 libevent 库和 libev 库能够大大 ...

  7. 体验了微信小程序,发现安卓用户终于把果粉“碾压”了一次

    今天早上,张小龙在微信公开课上分享了小程序的理念,并且公布了小程序将于1月9日上线. 为了体现张小龙对未来程序形态的理解,小程序有四个特定:无需安装.触手可及.用完即走.无需卸载.今天,36氪刚好有机 ...

  8. C++程序设计之四书五经[转自2004程序员杂志]--下篇

    C++程序设计之四书五经(下篇) 作者:荣耀 我在上篇中“盘点”了TCPL和D&E以及入门教程.高效和健壮编程.模板和泛型编程等方面共十几本C++好书.冬去春来,让我们继续C++书籍精彩之旅. ...

  9. Spring MVC 入门示例讲解

    在本例中,我们将使用Spring MVC框架构建一个入门级web应用程序.Spring MVC 是Spring框架最重要的的模块之一.它以强大的Spring IoC容器为基础,并充分利用容器的特性来简 ...

随机推荐

  1. 用Putty连接Linux

    随着linux应用的普及,linux管理越来越依赖远程管理.在各种telnet类工具中,putty是其中最出色的一个. 一.Putty简介     Putty是一个免费小巧的Win32平台下的teln ...

  2. 织梦在广告(myad)中使用css样式

    使用单引号,以及只有style这一个属性

  3. spring中bean的作用域属性single与prototype的区别

    https://blog.csdn.net/linwei_1029/article/details/18408363

  4. linux SPI驱动——spi core(四)

    一: SPI核心,就是指/drivers/spi/目录下spi.c文件中提供给其他文件的函数,首先看下spi核心的初始化函数spi_init(void). 1: static int __init s ...

  5. Linux内核源码分析方法_转

    Linux内核源码分析方法 转自:http://www.cnblogs.com/fanzhidongyzby/archive/2013/03/20/2970624.html 一.内核源码之我见 Lin ...

  6. js document.queryCommandState() 各个参数

    命令标识符 2D-Position 允许通过拖曳移动绝对定位的对象. AbsolutePosition 设定元素的 position 属性为“absolute”(绝对). BackColor 设置或获 ...

  7. 九度OJ 1171:C翻转 (矩阵计算)

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:4649 解决:1530 题目描述: 首先输入一个5 * 5的数组,然后输入一行,这一行有四个数,前两个代表操作类型,后两个数x y代表需操作 ...

  8. Kubernetes TensorFlow 默认 特定 集群管理器

    Our goal is to foster an ecosystem of components and tools that relieve the burden of running applic ...

  9. cocos2d-js添加百度MSSP插屏(通过jsb反射机制)

    1.导入jar包.... 2.修改AndroidManifest.xml文件 添加: <meta-data android:name="BaiduMobAd_APP_ID" ...

  10. thinkphp5 (最棒的php开源框架)

    tp5的唯一可访问目录是public,即项目根目录: http://localhost/tp5/public/ 开发规范: 类库.函数文件统一以.php为后缀 类(命名和路径)和命名空间保持一致 类文 ...