Nginx源码分析-connections数组

本文详细解析了Nginx如何通过connections数组维护连接,并利用链表形式高效地获取和释放连接槽位。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

转载申明:本文可以任意转载,但需注明原文地址,谢谢!

 

本文的标题让我纠结了好久,不知道是connections数组合适,还是connections链表更合适;Nginx在此或多或少的注入了二者的特点,先不管是叫数组还是叫链表吧,只要能够弄明白这个connections是怎么回事就大功告成。Nginx的每个worker进程都使用一个相同的connections数组来维护每个连接。当有一个新的连接建立时,Nginx需要从这个数组取出一个slot来存放这个连接;相反,有一个连接断开时,也需要将相应的slot归还给connections数组。本文就来看一下这个connections数组是如何构造的,能够让Nginx方便,快速的获取/释放一个slot。我从ngx_event_process_init()函数中拼凑了如下一段代码:

  1. /*分配数组*/  
  2. cycle->connections =  
  3.        ngx_alloc(sizeof(ngx_connection_t) * cycle->connection_n, cycle->log);  
  4.    if (cycle->connections == NULL) {  
  5.        return NGX_ERROR;  
  6.    }  
  7. c = cycle->connections;  
  8. i = cycle->connection_n;  
  9.    next = NULL;  
  10. /*将数组中的所有slot用指针串起来,形成一个链表*/  
  11.    do {  
  12.        i--;  
  13.        c[i].data = next;  
  14.        。。。。。。。。。。。。。。。。。  
  15.        next = &c[i];  
  16.    } while (i);  
  17. /*free_connections指向一个可用的slot*/  
  18.    cycle->free_connections = next;  
  19.    /*free_connection_n可用slot的个数*/  
  20.    cycle->free_connection_n = cycle->connection_n;  
 

经过上述代码过后,形成的数据结构如下图所示:

现在,Nginx已经为我们构造起了一个空的connections数组,我们需要建立一个新的连接,因此调用函数ngx_get_connection()从connections数组中获取一个可用的slot来维护这个连接。首先,我们看一下ngx_get_connection()函数的简化实现:

  1. ngx_connection_t *  
  2. ngx_get_connection(ngx_socket_t s, ngx_log_t *log)  
  3. {  
  4.     ngx_uint_t         instance;  
  5.     ngx_event_t       *rev, *wev;  
  6.     ngx_connection_t  *c;  
  7.     。。。。。。。。。。。。  
  8.       
  9.     /*从free_connecionts处就能够直接获得一个可用的slot*/  
  10.     c = ngx_cycle->free_connections;  
  11.     if (c == NULL) {  
  12.         return NULL;  
  13.     }  
  14.     /*移动free_connections指针到获取的C的下一个可用slot上*/  
  15.     ngx_cycle->free_connections = c->data;  
  16.     ngx_cycle->free_connection_n--;  
  17.     。。。。。。。。/*省略很多对c的操作*/  
  18.     return c;  
  19. }  
 

从刚建立的connections数组中获取一个slot后的,效果图如下:

在上述状态下,我们再获取一个slot,那么就得到如下的结果了:

到此,我们基本明白从connections数组中获取一个slot的操作过程了,接下来看看释放一个slot的过程是如何的。和ngx_get_connection()函数对应的是ngx_free_connection()函数。简化实现如下:

  1. void  
  2. ngx_free_connection(ngx_connection_t *c)  
  3. {  
  4.     /*规划的slot c指针可用的slot链表头*/  
  5.     c->data = ngx_cycle->free_connections;  
  6.     /*移动free_connections到c上*/  
  7.     ngx_cycle->free_connections = c;  
  8.     ngx_cycle->free_connection_n++;  
  9.     。。。。。。。。  
  10. }  
 

释放第一次获取的slot后的效果如下:

再释放第二次获取的slot后,又变成如下的效果了:

 

 

虽然最后整个connections数组都没有被使用,但和起初创建好的时候,还是有一定的区别的。但最重要的是——所有可以使用的slot都还是照样在一个链表上,只是slot的串联顺序不一样了。本文只是给出了一种很简单很简单的获取释放slot的case。你完全可以异想天开的获取释放slot的顺序,构造出看上去比较复杂的一个链表,但本质始终还是一样的。

 

一个好的数据结构,不光要高效,还得易用,Nginx在这个connection维护方面,做到了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值