http://cjhust.blog.163.com/blog/static/175827157201348112639361/

 

Hello world

模块功能:注册一个定时事件,每过一秒钟打印一次hello world。ngx_add_timer函数就是用来完成将一个新的定时事件加入定时器红黑树中,定时事件被执行后,就会从树中移除,因此要想不断的循环打印hello world,就需要在事件回调函数被调用后再将事件给添加到定时器红黑树中。 ngx_http_hello_process_init是注册在模块的进程初始化阶段的回调函数上。

static ngx_connection_t dummy;

static ngx_event_t ev;

static void

ngx_http_hello_print(ngx_event_t *ev)

{

printf("hello world\n");

ngx_add_timer(ev, 1000);

}

static ngx_int_t

ngx_http_hello_process_init(ngx_cycle_t *cycle)

{

dummy.fd = (ngx_socket_t) -1;

ngx_memzero(&ev, sizeof(ngx_event_t));

ev.handler = ngx_http_hello_print;

ev.log = cycle->log;

ev.data = &dummy;

ngx_add_timer(&ev, 1000);

return NGX_OK;

}

4、操作函数

ngx_event_timer_init

函数功能:完成定时器红黑树的建树操作。

ngx_int_t

ngx_event_timer_init(ngx_log_t *log)

{

ngx_rbtree_init(&ngx_event_timer_rbtree, &ngx_event_timer_sentinel,

ngx_rbtree_insert_timer_value);

#if (NGX_THREADS)

if (ngx_event_timer_mutex) {

ngx_event_timer_mutex->log = log;

return NGX_OK;

}

ngx_event_timer_mutex = ngx_mutex_init(log, 0);

if (ngx_event_timer_mutex == NULL) {

return NGX_ERROR;

}

#endif

return NGX_OK;

}

ngx_event_add_timer(ngx_event_t *ev, ngx_msec_t timer)

//timer就是一个int的值,表示超时的事件,用于表示红黑树节点的key

static ngx_inline void

ngx_event_add_timer(ngx_event_t *ev, ngx_msec_t timer)

{

ngx_msec_t      key;

ngx_msec_int_t  diff;

key = ngx_current_msec + timer;   //现在时间+超时时间=过期时间

//已经设置了定时器

if (ev->timer_set) {

/*

* Use a previous timer value if difference between it and a new

* value is less than NGX_TIMER_LAZY_DELAY milliseconds: this allows

* to minimize the rbtree operations for fast connections.

*/

diff = (ngx_msec_int_t) (key - ev->timer.key);

//2次比较相近,则忽略这次的

if (ngx_abs(diff) < NGX_TIMER_LAZY_DELAY) {  //300

ngx_log_debug3(NGX_LOG_DEBUG_EVENT, ev->log, 0,

"event timer: %d, old: %M, new: %M",

ngx_event_ident(ev->data), ev->timer.key, key);

return;

}

//否则,删除old

ngx_del_timer(ev);

}

ev->timer.key = key;

ngx_log_debug3(NGX_LOG_DEBUG_EVENT, ev->log, 0,

"event timer add: %d: %M:%M",

ngx_event_ident(ev->data), timer, ev->timer.key);

ngx_mutex_lock(ngx_event_timer_mutex);

//事件的timer域插入到红黑树当中

ngx_rbtree_insert(&ngx_event_timer_rbtree, &ev->timer);   // ev->timer是node

ngx_mutex_unlock(ngx_event_timer_mutex);

ev->timer_set = 1;

}

ngx_event_del_timer(ngx_event_t *ev)

static ngx_inline void

ngx_event_del_timer(ngx_event_t *ev)

{

ngx_log_debug2(NGX_LOG_DEBUG_EVENT, ev->log, 0,

"event timer del: %d: %M",

ngx_event_ident(ev->data), ev->timer.key);

ngx_mutex_lock(ngx_event_timer_mutex);

ngx_rbtree_delete(&ngx_event_timer_rbtree, &ev->timer);

ngx_mutex_unlock(ngx_event_timer_mutex);

#if (NGX_DEBUG)

ev->timer.left = NULL;

ev->timer.right = NULL;

ev->timer.parent = NULL;

#endif

ev->timer_set = 0;

}

ngx_event_find_timer(void)

函数功能:用于获取当前事件红黑树中最小的超时时间,将其提供给epoll,让epoll的wait在这个时间内唤醒。

ngx_msec_t

ngx_event_find_timer(void)

{

ngx_msec_int_t      timer;

ngx_rbtree_node_t  *node, *root, *sentinel;

if (ngx_event_timer_rbtree.root == &ngx_event_timer_sentinel) {

return NGX_TIMER_INFINITE;

}

ngx_mutex_lock(ngx_event_timer_mutex);

root = ngx_event_timer_rbtree.root;

sentinel = ngx_event_timer_rbtree.sentinel;

node = ngx_rbtree_min(root, sentinel);

ngx_mutex_unlock(ngx_event_timer_mutex);

timer = (ngx_msec_int_t) (node->key - ngx_current_msec);

return (ngx_msec_t) (timer > 0 ? timer : 0);

}

ngx_event_expire_timers(void)

函数功能:处理红黑树中的所有超时事件。

void

ngx_event_expire_timers(void)

{

ngx_event_t        *ev;

ngx_rbtree_node_t  *node, *root, *sentinel;

sentinel = ngx_event_timer_rbtree.sentinel;

//找到所有的超时的timer,然后处理它们

for ( ;; ) {

ngx_mutex_lock(ngx_event_timer_mutex);

root = ngx_event_timer_rbtree.root;

if (root == sentinel) {

return;

}

//获取key最小的节点

node = ngx_rbtree_min(root, sentinel);

//该节点是否超时

if ((ngx_msec_int_t) (node->key - ngx_current_msec) <= 0) {

ev = (ngx_event_t *) ((char *) node - offsetof(ngx_event_t, timer));

//类似于ngx_event_del_timer

ngx_rbtree_delete(&ngx_event_timer_rbtree, &ev->timer);

ngx_mutex_unlock(ngx_event_timer_mutex);

ev->timer_set = 0;

ev->timedout = 1;

ev->handler(ev);

continue;

}

break;

}

ngx_mutex_unlock(ngx_event_timer_mutex);

}

5、一个例子

ngx_process_events_and_timers(ngx_cycle_t *cycle)

void

ngx_process_events_and_timers(ngx_cycle_t *cycle)

{

ngx_uint_t  flags;

ngx_msec_t  timer, delta;

if (ngx_timer_resolution) {  //0

timer = NGX_TIMER_INFINITE;

flags = 0;

} else {

timer = ngx_event_find_timer();                        //返回的是最小的超时时间,单位是毫秒

flags = NGX_UPDATE_TIME;// 1

#if (NGX_THREADS)

if (timer == NGX_TIMER_INFINITE || timer > 500) {

timer = 500;

}

#endif

}

。。。

delta = ngx_current_msec;

//下面这个函数就是处理事件的函数(包括新连接建立事件),网络IO事件等等

(void) ngx_process_events(cycle, timer, flags);    //ngx_epoll_process_events

delta = ngx_current_msec - delta;                       //时间差,因为ngx_time_update已更新

。。。

if (delta) {

ngx_event_expire_timers();                           //处理所有的超时事件

}

。。。

}

nginx模块开发(31)—定时器模型的更多相关文章

  1. Nginx模块开发入门

    前言 Nginx是当前最流行的HTTP Server之一,根据W3Techs的统计,目前世界排名(根据Alexa)前100万的网站中,Nginx的占有率为6.8%.与Apache相比,Nginx在高并 ...

  2. [转] Nginx模块开发入门

    前言 Nginx是当前最流行的HTTP Server之一,根据W3Techs的统计,目前世界排名(根据Alexa)前100万的网站中,Nginx的占有率为6.8%.与Apache相比,Nginx在高并 ...

  3. Nginx模块开发入门(转)

    前言 Nginx是当前最流行的HTTP Server之一,根据W3Techs的统计,目前世界排名(根据Alexa)前100万的网站中,Nginx的占有率为6.8%.与Apache相比,Nginx在高并 ...

  4. Nginx模块开发入门(转)

    前言 Nginx是当前最流行的HTTP Server之一,根据W3Techs的统计,目前世界排名(根据Alexa)前100万的网站中,Nginx的占有率为6.8%.与Apache相比,Nginx在高并 ...

  5. FW: Nginx模块开发入门

    前言 Nginx是当前最流行的HTTP Server之一,根据W3Techs的统计,目前世界排名(根据Alexa)前100万的网站中,Nginx的占有率为6.8%.与Apache相比,Nginx在高并 ...

  6. 【转】Nginx模块开发入门

    转自: http://kb.cnblogs.com/page/98352/ 结论:对Nginx模块开发入门做了一个helloworld的示例,简单易懂.也有一定的深度.值得一看. Nginx模块开发入 ...

  7. 解剖Nginx·模块开发篇(1)跑起你的 Hello World 模块!

    1 学习 Nginx 模块开发需要有哪些准备? 需要的预备知识不多,有如下几点: 有过一些 C 语言的编程经历: 知道 Nginx 是干嘛的,并有过编写或改写 Nginx 的配置文件的经历. OK,就 ...

  8. linux下nginx模块开发入门

    本文模块编写参考http://blog.codinglabs.org/articles/intro-of-nginx-module-development.html 之前讲了nginx的安装,算是对n ...

  9. nginx模块开发篇 (阿里著作)

    背景介绍 nginx历史 使用简介 nginx特点介绍 nginx平台初探(100%) 初探nginx架构(100%) nginx基础概念(100%) connection request 基本数据结 ...

  10. 转:nginx模块开发——handler(一)

    handler模块简介 相信大家在看了前一章的模块概述以后,都对nginx的模块有了一个基本的认识.基本上作为第三方开发者最可能开发的就是三种类型的模块,即handler,filter和load-ba ...

随机推荐

  1. [Tip] 如何在BeyondCompare中忽略不重要的区别.

    在使用BeyondCompare时,有时需要忽略一些不重要的区别,下面的链接教你如何通过定义语法元素来实现这个功能. http://www.scootersoftware.com/support.ph ...

  2. mysql 学习碎片

    1.mysql 中执行 sql字符串 set @strSql='select 1200 as item'; prepare select_sql from @strSql; execute selec ...

  3. Debian 基本设置

    在此做一下记录,省得以后再老是去搜索了 我的debian版本v:8.1.0 我是在Hyper-V中装的,启动后一直ping不通网址,后来才知道Debian不能直接ping域名,只能ping IP... ...

  4. ArcGIS“一个或多个ActiveX控件无法显示...”问题的解决方案

    ArcMap启动时的一个警告信息“一个或多个ActiveX控件无法显示...”,如图 出现这种情况,有可能的原因是IE浏览器的安全选项设置被修改了.比如被手动修改过,或者被第三方系统杀毒优化软件修改了 ...

  5. c# File 操作

    //1.---------文件夹创建.移动.删除--------- //创建文件夹 Directory.CreateDirectory(Server.MapPath("a")); ...

  6. java中产生对象的两种方式

    /* * 普通new对象的过程! */ Person pp = new Person(); System.out.println(pp); /* * 利用代用参数的构造器产生对象实例! * 首先获得相 ...

  7. 【初识 JQMobile 小小总结】

    作为一个前端新手,之前还没有接触过手机端的项目.两周前项目经理告诉我要做手机端,让我用JQMobile. 之前在前端群里,偶尔听说过jqmobile很坑,自己又查了下其他框架,比如zepto.amaz ...

  8. Tools - Notepad++

    NotePad++ https://notepad-plus-plus.org/ 修改主题 依次点击设置---语言格式设置---选择主题,在显示界面中修改相关设置(背景色.前景色.字体等). 双文本横 ...

  9. 超好玩!10款神奇的字符图案 & 词汇云生成工具

    在这里,我们推荐10款惊人的字符图案生成工具.词云可以定义为词频的图形表示,而字符图案发生器是一个把数据,如文字和标签在以视觉和吸引人的方式展示的简单的工具.这些生成工具具有不同的功能,其中包括不同的 ...

  10. Android 学习笔记之网络通信基础+WebView....

    PS:加快学习进度...下周一完成Android网络通信...然后正式进入实战... 学习内容: 1.Android中Http基础... 2.Android中的Socket基础... 3.Androi ...