Nginx学习之五-nginx基础架构设计

一、优秀的模块化设计
(1)高度抽象的模块接口
所有的模块都遵循着同样的ngx_module_t接口设计规范。

(2)模块接口简单,灵活性高
模块的基本接口ngx_module_t足够简单,只涉及模块的初始化、退出以及对配置项的处理,这同时带来了足够的灵活性,是的nginx比较简单的实现了动态可修改性,也就是保持服务正常运行下使得系统功能发生改变。
102 struct ngx_module_s {
103     ngx_uint_t            ctx_index;//当前模块在某一类模块中的索引
104     ngx_uint_t            index;//当前模块在ngx_modules数组中的序号
105 
106     ngx_uint_t            spare0;
107     ngx_uint_t            spare1;
108     ngx_uint_t            spare2;
109     ngx_uint_t            spare3;
110 
111     ngx_uint_t            version;
112 
113     void                 *ctx;//指向一类模块的上下文结构体,特定类型模块的公共接口
114     ngx_command_t        *commands;//处理配置项
115     ngx_uint_t            type;//模块类型
116 
117     ngx_int_t           (*init_master)(ngx_log_t *log);//框架从来不会用到
118 
119     ngx_int_t           (*init_module)(ngx_cycle_t *cycle);//初始化所有模块时调用。在master/worker模式下,在启动子进程前完成
120 
121     ngx_int_t           (*init_process)(ngx_cycle_t *cycle);//正常服务器调用
122     ngx_int_t           (*init_thread)(ngx_cycle_t *cycle);
123     void                (*exit_thread)(ngx_cycle_t *cycle);
124     void                (*exit_process)(ngx_cycle_t *cycle);//服务停止前调用
125 
126     void                (*exit_master)(ngx_cycle_t *cycle);//master进程退出前调用
127 
128     uintptr_t             spare_hook0;
129     uintptr_t             spare_hook1;
130     uintptr_t             spare_hook2;
131     uintptr_t             spare_hook3;
132     uintptr_t             spare_hook4;
133     uintptr_t             spare_hook5;
134     uintptr_t             spare_hook6;
135     uintptr_t             spare_hook7;
136 };
上述代码中void*类型的成员ctx一般用于表示在不同类型的模块中一种类型模块所具备的通用性几接口。

(3)多层次、多类别的模块设计
Nginx常用模块机器之间的关系:


可以注意到,ngx_module_t接口中有一个type成员,指明了nginx允许在设计模块时定义模块类型这个概念。
配置模块的类型是NGX_CONF_MODULE,仅有一个模块。配置模块是所有模块的基础,它实现了最基本的配置项的解析功能(就是解析nginx.conf文件)。
nginx定义了一种基础类型的模块:核心模块,模块类型是NGX_CORE_MODULE。定义核心模块的目的是使得非模块化的框架代码只关注与如何调用6个核心模块。
事件模块、HTTP模块、mail模块这三种模块的共性是:实际上它们在核心模块中各有1个模块作为自己的代言人,并在同类模块中有1个作为核心业务与管理功能的模块。
事件模块是HTTP模块和mail模块的基础。



二、事件驱动架构
nginx采用完全的事件驱动架构来处理业务,这与传统的web服务器(如Apache)是不同的。
对于传统web服务器而言,采用的所谓事件驱动往往局限在tcp连接建立、关闭事件上。一个连接建立后,在其关闭之前的所有操作都不再是事件驱动,这时会退化成按序执行每个操作的批处理模式,这样每个请求在连接建立后都将始终占用着系统资源,直到连接关闭后才会释放资源。
这种传统的web服务器往往把一个进程或线程作为事件消费者,当一个请求产生的事件被该进程处理时,直到这个请求处理结束时进程资源都将被这一个请求所占用。

Nginx则不然,不会使用进程或线程来作为事件消费者,所谓的事件消费者只能是某个模块(在这里没有进程的概念)。只有事件收集、分发器才有资格占用进程资源,它们会在分发某个事件时调用事件消费模块使用当前占用的资源。

Nginx处理事件的简单模型如下图:


从上面的内容可一看出传统web服务器与nginx之间的重要差别:前者是每个事件消费者独占一个进程资源,后者的事件消费者只是被事件分发者短期调用而已。这种设计使得网络性能、用户感知的请求时延都得到提升,每个用户的请求都所产生的事件都得到及时响应,整个服务器的网络吞吐量都会由于事件的及时响应而增大。但这也会带来一个问题:事件消费者都不能有阻塞行为,否则由于长时间占用事件分发者进程而导致其它事件得不到及时响应。

三、请求的多阶段异步处理

请求的多阶段异步处理是基于事件(也只能基于事件)驱动架构实现。就是把一个请求的处理过程按照事件的触发方式分为多个节点,每个阶段都由事件收集、分发器来触发。

其优势:
这种设计配合事件驱动架构,将极大地提高网络的性能,同时使得每个进程全力运转,不会或者尽量少的出现进程休眠状况。因为一旦出现进程休眠,必然减少并发处理事件的数目,一定会降低网络性能,同时会增加请求处理时间的平均时延。这时,网络性能无法满足业务需求将只能增加进程数目,进程数目过多就会增加操作系统内核的额外操作:进程间切换。频繁的进程切换仍会消耗CPU等资源,从而降低网络性能。同时,休眠的进程会使进程占用的内存得不到有效释放,这最终必然导致系统可用内存的下降,从而影响系统能够处理的最大并发连接数。

参考资料:
《深入理解Nginx》


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值