Libev学习笔记4
这一节首先分析Libev的定时器部分,然后分析signal部分。
对定时器的使用主要有两个函数:
ev_timer_init (&timeout_watcher, timeout_cb, 5.5, .);
ev_timer_start (loop, &timeout_watcher);
和ev_io类型的watcher类似,timeout_watcher是一个类型为ev_timer的watcher,上面的ev_timer_init函数将它设置为5.5秒之后调用回调函数timeout_cb,最后一个参数0表示定时器不重复超时,执行完一次回调函数后就停止计时,如果最后一个参数为非0,那么回调函数第一次执行完之后,每个指定秒后回调函数会被重复执行。将ev_timer结构体展开:
typedef struct ev_timer
{
int active; /* private */ \ /* 非0表示watcher为激活状态,是periodics或timers数组的下标 */
int pending; /* private */ \ /* 非0表示watcher中有事件被触发,是pendings数组的下标 */
EV_DECL_PRIORITY /* private */ \ /* watcher的优先级 */
EV_COMMON /* rw */ \ /* void *data; */
EV_CB_DECLARE (type) /* private */ /* void (*cb)(EV_P_ struct type *w, int revents); */
ev_tstamp at; ev_tstamp repeat; /* rw */
} ev_timer;
其中at成员对应倒数第二个参数,表示多少秒后第一次触发超时;repeat成员对应最后一个参数,表示第一次超时触发之后每个多少秒重复触发。ev_timer_init函数实质上就是设置上面这些成员。ev_timer_start主要工作是将timer放入最小堆中,由最小堆统一管理所有的timer,代码如下:
/* 计算定时器的绝对触发事件并放入堆中 */
void noinline
ev_timer_start (EV_P_ ev_timer *w) EV_THROW
{
if (expect_false (ev_is_active (w)))
return; ev_at (w) += mn_now; /* 从现在开始经过at秒后触发 */++timercnt;
ev_start (EV_A_ (W)w, timercnt + HEAP0 - ); /* 将定时器w放入timers堆中 */
array_needsize (ANHE, timers, timermax, ev_active (w) + , EMPTY2);
ANHE_w (timers [ev_active (w)]) = (WT)w;
ANHE_at_cache (timers [ev_active (w)]); /* 更新heap */
upheap (timers, ev_active (w));
}
代码首先根据当前时间计算timer超时时刻的绝对时间,然后增加loop管理的timer个数timercnt,修改一些标志变量,最后把timer放入最小堆timers中,timers使用数组实现的一个最小堆,堆顶为离当前最近的一个timer。
timer定制完毕后就可以调用ev_run函数开始event loop了。在ev_run函数中,关于timer的代码流程大致如下:
int
ev_run (EV_P_ int flags)
{
.... do
{
....
/* 计算epoll等事件驱动机制的阻塞时间 */
{/* 堆顶取出ANHE,减去当前时间,赋值给waittime */
/* 调整waittime */ backend_poll (EV_A_ waittime); /* epoll_poll() */
}
....
/* queue pending timers and reschedule them */
/* 调整堆,取出所有超时的timer */
timers_reify (EV_A); /* relative timers called last */
/* 调用pendings数组中watcher的回调函数 */
EV_INVOKE_PENDING;
}
}
根据最小堆timers计算事件驱动机制的阻塞等待时间,阻塞返回后timers_reify函数记录所有超时的timer,最后由EV_INVOKE_PENDING执行这些timer对应的超时回调函数。在timers_reify函数中,如果timer是单次触发类型,那么会把该timer从最小堆中删除,如果timer是重复触发类型,那么会把该timer重新放入最小堆中,等待下次触发。
接下来分析信号部分。Libev将对信号的处理融入到了I/O事件的处理当中。官方文档给出了一个关于使用signal的例子:
static void
sigint_cb (struct ev_loop *loop, ev_signal *w, int revents)
{
ev_break (loop, EVBREAK_ALL);
} ev_signal signal_watcher;
ev_signal_init (&signal_watcher, sigint_cb, SIGINT);
ev_signal_start (loop, &signal_watcher);
ev_signal是专门监控是否有指定信号发生的watcher。在上面的例子中,当有SIGINT信号发生时,执行回调函数sigint_cb。ev_signal_init就是初始化ev_signal结构体,包括保存回调函数,保存需要等待的信号等。之后调用ev_signal_start函数,该函数是理解Libev中信号处理机制的关键,它的关键代码如下:
void noinline
ev_signal_start (EV_P_ ev_signal *w) EV_THROW
{
signals [w->signum - ].loop = EV_A; ev_start (EV_A_ (W)w, );
wlist_add (&signals [w->signum - ].head, (WL)w); if (!((WL)w)->next)
{
struct sigaction sa; evpipe_init (EV_A); sa.sa_handler = ev_sighandler; /* 设置信号处理程序 */
sigfillset (&sa.sa_mask);
sa.sa_flags = SA_RESTART; /* if restarting works we save one iteration */
sigaction (w->signum, &sa, );
}
}
其中,signals是一个数组,数组元素类型为ANSIG,它代表信号,数组中的一个元素代表Libev需要监控的一个信号。在Linux中,信号值是一个整数,那么用信号值作为下标,就能很快找到对应的ANSIG结构体了。ANSIG结构体定义如下:
/* 一个信号对应一个ANSIG */
typedef struct
{
EV_ATOMIC_T pending; /* 1表示收到信号 */
#if EV_MULTIPLICITY
EV_P; /* 信号所属的loop */
#endif
WL head; /* 多个watcher可以同时检测一个信号,用该成员作链表头 */
} ANSIG;
所以,ev_signal_start函数主要做了如下几件事:
- 在signals数组中建立一个新的需要监控的ANSIG,并设置ANSIG中的成员;
- 设置ev_signal这个watcher中的一些成员变量;
- 调用evpipe_init函数,该函数中会生成一个pipe,evpipe[0]用作读,evpipe[1]用作写,然后用ev_io监控evpipe[0];
- 使用sigaction系统调用设置Linux中标准的信号处理函数为ev_sighandler,信号处理函数中往evpipe[1]写入一个字节。
那么整理一下流程,当系统接收到指定信号后,可以得出如下一系列步骤:
- 捕捉信号
- 信号处理函数ev_sighandler被调用
- 在信号处理函数中向evpipe[1]写数据
- evpipe[0]变为可读
- 触发包裹它的pipe_w事件
- pipe_w事件对应的可读回调函数pipecb被调用
- pipecb函数首先会读取evpipe[0],然后根据信号值获得对应的ANSIG进而获得watcher,最后将该watcher放入loop->pendings数组中
- 该watcher对应的用户回调函数会在稍后被调用
以上就是Libev处理信号的大致流程。可以看出,对信号的监控最后都同一到了对文件描述符的监控,也就是最终使用到了ev_io类型的watcher。
参考:
http://my.oschina.net/u/917596/blog/177300
http://www.cnblogs.com/foxmailed/archive/2013/02/04/2891077.html
Libev学习笔记4的更多相关文章
- libev学习笔记
转 libev的使用--结合Socket编程 作者:cxy450019566 之前自己学过一些libev编程的基础,这次写压测刚好用上了,才算真正动手写了些东西,在这里做一些总结.写这篇文章是为了用浅 ...
- Libev学习笔记3
设置完需要监听的事件之后,就开始event loop了.在Libev中,该工作由ev_run函数完成.它的大致流程如下: int ev_run (EV_P_ int flags) { do { /* ...
- Libev学习笔记2
这一节根据官方文档给出的简单示例,深入代码内部,了解其实现机制.示例代码如下: int main (void) { struct ev_loop *loop = EV_DEFAULT; ev_io_i ...
- Libev学习笔记1
和Libevent相似,Libev是一个高性事件驱动框架,据说性能比Libevent要高,bug比Libevent要少.Libev只是一个事件驱动框架,不是网络库,因为它的内部并没有任何socket编 ...
- libev 学习使用
libev 简单的I/O库. a high performance full featured event loop written in c libev 的大小也比 libevent 小得多并且自 ...
- js学习笔记:webpack基础入门(一)
之前听说过webpack,今天想正式的接触一下,先跟着webpack的官方用户指南走: 在这里有: 如何安装webpack 如何使用webpack 如何使用loader 如何使用webpack的开发者 ...
- PHP-自定义模板-学习笔记
1. 开始 这几天,看了李炎恢老师的<PHP第二季度视频>中的“章节7:创建TPL自定义模板”,做一个学习笔记,通过绘制架构图.UML类图和思维导图,来对加深理解. 2. 整体架构图 ...
- PHP-会员登录与注册例子解析-学习笔记
1.开始 最近开始学习李炎恢老师的<PHP第二季度视频>中的“章节5:使用OOP注册会员”,做一个学习笔记,通过绘制基本页面流程和UML类图,来对加深理解. 2.基本页面流程 3.通过UM ...
- 2014年暑假c#学习笔记目录
2014年暑假c#学习笔记 一.C#编程基础 1. c#编程基础之枚举 2. c#编程基础之函数可变参数 3. c#编程基础之字符串基础 4. c#编程基础之字符串函数 5.c#编程基础之ref.ou ...
随机推荐
- cocostudio导出plist文件
今天在用Armature类时用到cocostudio导出文件,由于美术的原因他使用的是中文命名法(这你敢相信),后面在导入程序中跟了下代码发现是解析plist文件有误,我就来比较正常功能文件和有错文件 ...
- netty中实现客户端首次连接绑定并非每次read检查的方法
需求场景 客户端第一次连接时,将客户端存起来 重写 ChannelHandlerAdapter 的 handlerAdded 方法
- python 学习之Windows 下的编码处理!
问题1: Non-ASCII character '\xe9' in file 问题原因:程序编码上出现问题 解决方法:在程序头部加上代码 #-*- coding: UTF-8 -*- 设置代码编码为 ...
- 尼康D5100使用设置及技巧,同样也适用尼康D5200
尼康D5100使用设置及技巧,同样也适用尼康D5200,希望对新手能有点帮助. 一.设置 1.优化校准:可以在menu菜单中找到它,一般使用"标准"就可以,建议将"标准& ...
- JavaScript 鸭子模型
Javascript:由 “鸭子类型” 得出来的推论 背景 学动态语言的都知道一句话:“如果它走起来像鸭子,而且叫起来像鸭子,那么它就是鸭子”,Javascript也支持鸭子类型,下文就说说鸭子类型在 ...
- Android 4.0 ProGuard 代码混淆 以及 proguard returned with error code 1.See console异常的解决方法
最近呢说要上线,就去找了下上线的方法...之前做过代码混淆,用的是progarud.cfg,但是呢自己反编译了之后还是无效,然后就丢着先不管了,因为实在不知道什么情况.今天来上线的时候结果总是报错,总 ...
- css 样式
<!doctype html> <html lang="en"> <head> <meta name="Generator&qu ...
- 关于tableView的简单实例
关于tableCell选中颜色 //无色 cell.selectionStyle = UITableViewCellSelectionStyleNone; //蓝色 cell.selectionSty ...
- HTTP初步注解
搜集了一下网上的资源和自己看过的一些书,小小总结了一波HTTP,现在也只是很肤浅的了解,期望以后深入理解后能写出更有营养的笔记. HTTP协议的主要特点 + 支持客户/服务器模式.+ 简单快速:客户向 ...
- 对Devexpress ASP.NET组件的一些看法
使用.net开发的应该都熟悉DevExpress这套组件,强大的功能,显著提高开发效率和提升用户体验. 不过好像大都用winform, 说起用asp.net组件来开发webform,很多人开口就说慢, ...