http://www.cnblogs.com/sld666666/archive/2010/05/05/1728417.html
首先说下感受吧。
这个是对nginx的main()函数的简单的分析了,大致看了下nginx.c的代码,发现有如下几个特点:
一:高度模块化,唯一把多个模块串起来的就是ngx_cycle_s这个结构体。
二:重新定义,不管是数据类型,还是库函数,还是系统函数,其都给它们包装了下,
unix网络编程里面就提倡这种做法,并把之称之为包裹函数。
三:注释太少,虽说里面有些代码是自明的,但是并不是所有的代码都能直接表现出其意义。
也许是我水平不太高的原因吧。
四:代码干净,简洁干练,这个只是我的主观的观点,呵呵,没有论证。
在这里,只是总体浏览了一下,没深入到代码细节上,如果以上或以下有错误的观点,望指教。
Let's begin!
在开始分析main函数的代码以前必 须了解一下一个典型的服务器的运行方式。
一个简单而有代表性的服务器程序的伪代码如下:
代码
struct sockaddr_in servaddr cliaddr;
int listenfd = socket(AF_INET, SOCK_STREAM,0);
bind(listenfd, (struct sockaddr_in*)(&servaddr), sizeof(servaddr));
listen(listenfd, 22);
for (;;) {
int clielen = sizeof(cliaddr);
int connfd = accept(listenfd, (strct sockaddr_in)(&cliaddr), & clielen );
if( 0 == fork())
I_will_do_something();
}
nginx作为一个http服务器当然不能免俗了。整个mian()算上空行200行左右,有效代码当然更少了。
好吧,现在来看看mian函数干了什么了。
main(int argc, char *const *argv)
{
ngx_int_t i;
ngx_log_t *log;
ngx_cycle_t *cycle, init_cycle;
ngx_core_conf_t *ccf;
ngx_get_options(argc, argv); /*根据参数,选定一实现什么功能,在此函数里面有对几个全局变量的 处理*/
if (ngx_show_version) {/*这个ngx_show_version就是函数ngx_get_options的处理结果了*/
I_will_show_information;
}
ngx_time_init();/*服务器时间的初始化*/
ngx_pid = ngx_getpid();/*得到进程id*/
log = ngx_log_init(ngx_prefix);/*服务器怎么能不写log呢,初始化之*/
ngx_memzero(&init_cycle, sizeof(ngx_cycle_t));/*meset(init_cycle, 0 , sizeof(ngx_cycle_t))*/
init_cycle.log = log;
ngx_cycle = &init_cycle;
ngx_pagesize = 1024;
init_cycle.pool = ngx_create_pool(1024, log);
/*初始化init_cycle, init_cycle 是一个非常庞大struct,里面包含了内存池,log,array,list,event,套节字,主机名 等等一系列信息*/
ngx_add_inherited_sockets(&init_cycle);/*这里就是前面说的服务器接听套节字过程的封装了*/
ngx_os_status(cycle->log);/*还是写log*/
ngx_cycle = cycle;
ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module);
ngx_create_pidfile()/*还是往文件写log信息*/
/*这里省下几个错误处理的动作*/
ngx_single_process_cycle(cycle)/*我要处理东西了*/
return 0;
}