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. WPF Litbox样式和模板

    1.在项目中使用ListBox时,经常会将ItemContainerStyle和ItemTemplate的作用搞混,ItemTemplate可以搞定一切好似ItemContainerStyle有点多余 ...

  2. JS调用WebService

    <html> <title>接口测试</title> <meta http-equiv="Content-Type" content=&q ...

  3. C# WinForm程序打印条码 Code39码1

    做WinForm程序需要打印条码,为了偷懒不想自己写生成条码的程序在网上下载一个标准的39码的字体,在程序里面换上这个条码字体即可打印条码了. 最重要的一点作为记录: 如果想把“123456789”转 ...

  4. jqGrid标题行与第一行之间有很大空白的问题解决。

    如题的问题,网上找了很久,都没有解决方案.最后发现,问题不在jqgrid的配置代码,问题在前台HTML代码. <table id="grid" height="30 ...

  5. ASP.NET MVC 4 Web编程

    http://spu.jd.com/11309606.html 第1章 入门第2章 控制器第3章 视图第4章 模型第5章 表单和HTML辅助方法第6章 数据注解和验证第7章 成员资格.授权和安全性第8 ...

  6. Linux高级编程--07.进程间通信

    每个进程各自有不同的用户地址空间,进程之间要交换数据必须通过在内核中开辟缓冲区,从而实现数据共享. 管道 管道是一种最基本的IPC机制,由pipe函数创建: int pipe(int filedes[ ...

  7. EF相关知识

    一.EF控制台命令: 1.enable-migrations -EnableAutomaticMigration -Force:使用自动迁移 2.add-migration "Country ...

  8. 标志数在wordcount程序中的应用与拓展

    wordcount程序要求测出文本中的单词数,字符数和行数. 设计思路: 将文件读入,逐字检测,检测到空格单词数加一,检测到回车行数单词数加一,如果既不是回车也不是空格则说明是字符,字符数加一 编程时 ...

  9. Sapi 添加语法的文章(转载)

    最近在做SAPI方面的工作,比较详细的中文资料不多,遇到各种问题,本来想着做完了项目总结一下,今天看到这篇文章,对于SAPI加载识别语法方面的描述十分详细,先转过来做个备份,谢谢原文博主:djyang ...

  10. Hekaton是如何影响你数据库的目标恢复时间(RTO)的

    这个周末我发现了SQL Server 2014里Hekaton的一个有趣副作用,很遗憾它会负面影响你数据库的目标恢复时间(Recovery Time Objective,RTO).你已知道,对于每个本 ...