nginx链表ngx_list_t

2024-05-04 22:32
文章标签 链表 nginx list ngx

本文主要是介绍nginx链表ngx_list_t,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本博客(http://blog.csdn.net/livelylittlefish )贴出作者(阿波)相关研究、学习内容所做的笔记,欢迎广大朋友指正!

【注】继续转载

Content

1.链表结构

1.2 ngx_list_t的逻辑结构

2.1创建链表

3.一个例子

3.2如何编译

4.小结


0. 序

 

本文继续介绍nginx的容器——链表。

链表实现文件:文件:./src/core/ngx_list.h/.c。.表示nginx-1.0.4代码目录,本文为/usr/src/nginx-1.0.4。

1. 链表结构

 

1.1 ngx_list_t结构

 

nginx的链表(头)结构为ngx_list_t,链表节点结构为ngx_list_part_t,定义如下。

[cpp]  view plain copy
  1. typedef struct ngx_list_part_s ngx_list_part_t;  
  2.    
  3. struct ngx_list_part_s {      //链表节点结构  
  4.     void             *elts;   //指向该节点实际的数据区(该数据区中可以存放nalloc个大小为size的元素)  
  5.     ngx_uint_t        nelts;  //实际存放的元素个数  
  6.     ngx_list_part_t  *next;   //指向下一个节点  
  7. };  
  8.    
  9. typedef struct{              //链表头结构  
  10.     ngx_list_part_t  *last;   //指向链表最后一个节点(part)  
  11.     ngx_list_part_t   part;   //链表头中包含的第一个节点(part)  
  12.     size_t            size;   //每个元素大小  
  13.     ngx_uint_t        nalloc; //链表所含空间个数,即实际分配的小空间的个数  
  14.     ngx_pool_t       *pool;   //该链表节点空间在此内存池中分配  
  15. }ngx_list_t;  


其中,sizeof(ngx_list_t)=28,sizeof(ngx_list_part_t)=12。

 

由此可见,nginx的链表也要从内存池中分配。对于每一个节点(list part)将分配nalloc个大小为size的小空间,实际分配的大小为(nalloc * size)。详见下文的分析。

 

1.2 ngx_list_t的逻辑结构

 

ngx_list_t结构引用了ngx_pool_t结构,因此本文参考nginx-1.0.4源码分析—内存池结构ngx_pool_t及内存管理一文画出相关结构的逻辑图,如下。注:本文采用UML的方式画出该图。

 

2. 链表操作

 

链表操作共3个,如下。

[cpp]  view plain copy
  1. //创建链表  
  2. ngx_list_t*ngx_list_create(ngx_pool_t *pool, ngx_uint_t n, size_t size);  
  3.    
  4. //初始化链表  
  5. static ngx_inline ngx_int_t ngx_list_init(ngx_list_t *list, ngx_pool_t *pool,  
  6.     ngx_uint_tn, size_t size);  
  7.    
  8. //添加元素  
  9. void*ngx_list_push(ngx_list_t *l)  

他们的实现都很简单,本文只分析创建链表和添加元素操作。

 

2.1创建链表

 

创建链表的操作实现如下,首先分配链表头(28B),然后分配头节点(即链表头中包含的part)数据区,两次分配均在传入的内存池(pool指向的内存池)中进行。然后简单初始化链表头并返回链表头的起始位置。

[cpp]  view plain copy
  1. ngx_list_t *  
  2. ngx_list_create(ngx_pool_t*pool, ngx_uint_t n, size_t size)  
  3. {  
  4.     ngx_list_t *list;  
  5.    
  6.     list = ngx_palloc(pool,sizeof(ngx_list_t));  //从内存池中分配链表头  
  7.     if (list == NULL) {  
  8.         return NULL;  
  9.     }  
  10.    
  11.     list->part.elts =ngx_palloc(pool, n * size); //接着分配n*size大小的区域作为链表数据区  
  12.     if (list->part.elts == NULL) {  
  13.         return NULL;  
  14.     }  
  15.    
  16.     list->part.nelts = 0;     //初始化  
  17.     list->part.next = NULL;  
  18.     list->last = &list->part;  
  19.     list->size = size;  
  20.     list->nalloc = n;  
  21.     list->pool = pool;  
  22.    
  23.     return list;    //返回链表头的起始位置  
  24. }  
创建链表后内存池的物理结构图如下。


2.2添加元素

 

添加元素操作实现如下,同nginx数组实现类似,其实际的添加操作并不在该函数中完成。函数ngx_list_push返回可以在该链表数据区中放置元素(元素可以是1个或多个)的位置,而添加操作即在获得添加位置之后进行,如后文的例子。

[cpp]  view plain copy
  1. void *  
  2. ngx_list_push(ngx_list_t*l)  
  3. {  
  4.     void             *elt;  
  5.     ngx_list_part_t  *last;  
  6.    
  7.     last = l->last;  
  8.    
  9.     if (last->nelts ==l->nalloc) {  //链表数据区满  
  10.    
  11.         /* the last part is full, allocate anew list part */  
  12.    
  13.         last =ngx_palloc(l->pool, sizeof(ngx_list_part_t));  //分配节点(list part)  
  14.         if (last == NULL) {  
  15.             return NULL;  
  16.         }  
  17.    
  18.         last->elts =ngx_palloc(l->pool, l->nalloc * l->size);//分配该节点(part)的数据区  
  19.         if (last->elts == NULL) {  
  20.             return NULL;  
  21.         }  
  22.    
  23.         last->nelts = 0;  
  24.         last->next = NULL;  
  25.    
  26.         l->last->next =last;  //将分配的list part插入链表  
  27.         l->last = last;        //并修改list头的last指针  
  28.     }  
  29.    
  30.     elt = (char *)last->elts + l->size * last->nelts; //计算下一个数据在链表数据区中的位置  
  31.     last->nelts++;  //实际存放的数据个数加1  
  32.    
  33.     return elt;  //返回该位置  
  34. }  

由此可见,向链表中添加元素实际上就是从内存池中分配链表节点(part)及其该节点的实际数据区,并修改链表节点(part)信息。

 

注1:与数组的区别,数组数据区满时要扩充数据区空间;而链表每次要分配节点及其数据区。

注2:链表的每个节点(part)的数据区中可以放置1个或多个元素,这里的元素可以是一个整数,也可以是一个结构。

 

下图是一个有3个节点的链表的逻辑结构图。

 

图中的线太多,容易眼晕,下面这个图可能好一些。

 

3. 一个例子

 

理解并掌握开源软件的最好方式莫过于自己写一些测试代码,或者改写软件本身,并进行调试来进一步理解开源软件的原理和设计方法。本节给出一个创建内存池并从中分配一个链表的简单例子。在该例中,链表的每个节点(part)可存放5个元素,每个元素4字节大小,创建链表后,要向链表添加15个整型元素。

 

3.1代码

[cpp]  view plain copy
  1. /** 
  2.  * ngx_list_t test, to test ngx_list_create, ngx_list_push 
  3.  */  
  4.   
  5. #include <stdio.h>  
  6. #include "ngx_config.h"  
  7. #include "ngx_conf_file.h"  
  8. #include "nginx.h"  
  9. #include "ngx_core.h"  
  10. #include "ngx_string.h"  
  11. #include "ngx_palloc.h"  
  12. #include "ngx_list.h"  
  13.   
  14. volatile ngx_cycle_t  *ngx_cycle;  
  15.   
  16. void ngx_log_error_core(ngx_uint_t level, ngx_log_t *log, ngx_err_t err,  
  17.             const char *fmt, ...)  
  18. {  
  19. }  
  20.   
  21. void dump_pool(ngx_pool_t* pool)  
  22. {  
  23.     while (pool)  
  24.     {  
  25.         printf("pool = 0x%x\n", pool);  
  26.         printf("  .d\n");  
  27.         printf("    .last = 0x%x\n", pool->d.last);  
  28.         printf("    .end = 0x%x\n", pool->d.end);  
  29.         printf("    .next = 0x%x\n", pool->d.next);  
  30.         printf("    .failed = %d\n", pool->d.failed);  
  31.         printf("  .max = %d\n", pool->max);  
  32.         printf("  .current = 0x%x\n", pool->current);  
  33.         printf("  .chain = 0x%x\n", pool->chain);  
  34.         printf("  .large = 0x%x\n", pool->large);  
  35.         printf("  .cleanup = 0x%x\n", pool->cleanup);  
  36.         printf("  .log = 0x%x\n", pool->log);  
  37.         printf("available pool memory = %d\n\n", pool->d.end - pool->d.last);  
  38.         pool = pool->d.next;  
  39.     }  
  40. }  
  41.   
  42. void dump_list_part(ngx_list_t* list, ngx_list_part_t* part)  
  43. {  
  44.     int *ptr = (int*)(part->elts);  
  45.     int loop = 0;  
  46.   
  47.     printf("  .part = 0x%x\n", &(list->part));  
  48.     printf("    .elts = 0x%x  ", part->elts);  
  49.     printf("(");  
  50.     for (; loop < list->nalloc - 1; loop++)  
  51.     {  
  52.         printf("0x%x, ", ptr[loop]);  
  53.     }  
  54.     printf("0x%x)\n", ptr[loop]);  
  55.     printf("    .nelts = %d\n", part->nelts);  
  56.     printf("    .next = 0x%x", part->next);  
  57.     if (part->next)  
  58.         printf(" -->\n");  
  59.     printf(" \n");  
  60. }  
  61.   
  62. void dump_list(ngx_list_t* list)  
  63. {  
  64.     if (list == NULL)  
  65.         return;  
  66.   
  67.     printf("list = 0x%x\n", list);  
  68.     printf("  .last = 0x%x\n", list->last);  
  69.     printf("  .part = 0x%x\n", &(list->part));  
  70.     printf("  .size = %d\n", list->size);  
  71.     printf("  .nalloc = %d\n", list->nalloc);  
  72.     printf("  .pool = 0x%x\n\n", list->pool);  
  73.   
  74.     printf("elements:\n");  
  75.   
  76.     ngx_list_part_t *part = &(list->part);  
  77.     while (part)  
  78.     {  
  79.         dump_list_part(list, part);  
  80.         part = part->next;  
  81.     }  
  82.     printf("\n");  
  83. }  
  84.   
  85. int main()  
  86. {  
  87.     ngx_pool_t *pool;  
  88.     int i;  
  89.   
  90.     printf("--------------------------------\n");  
  91.     printf("create a new pool:\n");  
  92.     printf("--------------------------------\n");  
  93.     pool = ngx_create_pool(1024, NULL);  
  94.     dump_pool(pool);  
  95.   
  96.     printf("--------------------------------\n");  
  97.     printf("alloc an list from the pool:\n");  
  98.     printf("--------------------------------\n");  
  99.     ngx_list_t *list = ngx_list_create(pool, 5, sizeof(int));  
  100.     dump_pool(pool);  
  101.   
  102.     for (i = 0; i < 15; i++)  
  103.     {  
  104.         int *ptr = ngx_list_push(list);  
  105.         *ptr = i + 1;  
  106.     }  
  107.   
  108.     printf("--------------------------------\n");  
  109.     printf("the list information:\n");  
  110.     printf("--------------------------------\n");  
  111.     dump_list(list);  
  112.   
  113.     printf("--------------------------------\n");  
  114.     printf("the pool at the end:\n");  
  115.     printf("--------------------------------\n");  
  116.     dump_pool(pool);  
  117.   
  118.     ngx_destroy_pool(pool);  
  119.     return 0;  
  120. }  

3.2如何编译

 

请参考nginx-1.0.4源码分析—内存池结构ngx_pool_t及内存管理一文。本文编写的makefile文件如下。

[cpp]  view plain copy
  1. CXX = gcc  
  2. CXXFLAGS +=-g -Wall -Wextra  
  3.    
  4. NGX_ROOT =/usr/src/nginx-1.0.4  
  5.    
  6. TARGETS =ngx_list_t_test  
  7. TARGETS_C_FILE= $(TARGETS).c  
  8.    
  9. CLEANUP = rm-f $(TARGETS) *.o  
  10.    
  11. all:$(TARGETS)  
  12.    
  13. clean:  
  14. $(CLEANUP)  
  15.    
  16. CORE_INCS =-I. \  
  17. -I$(NGX_ROOT)/src/core \  
  18. -I$(NGX_ROOT)/src/event \  
  19. -I$(NGX_ROOT)/src/event/modules \  
  20. -I$(NGX_ROOT)/src/os/unix \  
  21. -I$(NGX_ROOT)/objs \  
  22.    
  23. NGX_PALLOC =$(NGX_ROOT)/objs/src/core/ngx_palloc.o  
  24. NGX_STRING =$(NGX_ROOT)/objs/src/core/ngx_string.o  
  25. NGX_ALLOC =$(NGX_ROOT)/objs/src/os/unix/ngx_alloc.o  
  26. NGX_LIST =$(NGX_ROOT)/objs/src/core/ngx_list.o  
  27.    
  28. $(TARGETS):$(TARGETS_C_FILE)  
  29. $(CXX) $(CXXFLAGS) $(CORE_INCS) $(NGX_PALLOC) $(NGX_STRING)$(NGX_ALLOC) $(NGX_LIST) $^ -o $@  

3.3运行结果

[plain]  view plain copy
  1. # ./ngx_list_t_test  
  2. -------------------------------- create a new pool:  
  3. -------------------------------- pool = 0x9208020 .d .last = 0x9208048  
  4.     .end = 0x9208420  
  5.     .next = 0x0  
  6.     .failed = 0 .max = 984  
  7.   .current = 0x9208020  
  8.   .chain = 0x0  
  9.   .large = 0x0  
  10.   .cleanup = 0x0  
  11.   .log = 0x0 available pool memory = 984  
  12. -------------------------------- alloc an list from the pool:  
  13. -------------------------------- pool = 0x9208020 .d .last = 0x9208078  
  14.     .end = 0x9208420  
  15.     .next = 0x0  
  16.     .failed = 0 .max = 984  
  17.   .current = 0x9208020  
  18.   .chain = 0x0  
  19.   .large = 0x0  
  20.   .cleanup = 0x0  
  21.   .log = 0x0 available pool memory = 936  
  22. -------------------------------- the list information:  
  23. -------------------------------- list = 0x9208048 .last = 0x9208098  
  24.   .part = 0x920804c  
  25.   .size = 4  
  26.   .nalloc = 5  
  27.   .pool = 0x9208020  
  28. elements: .part = 0x920804c .elts = 0x9208064  (0x1, 0x2, 0x3, 0x4, 0x5)  
  29.     .nelts = 5  
  30.     .next = 0x9208078 -->  
  31.   .part = 0x920804c .elts = 0x9208084  (0x6, 0x7, 0x8, 0x9, 0xa)  
  32.     .nelts = 5  
  33.     .next = 0x9208098 -->  
  34.   .part = 0x920804c .elts = 0x92080a4  (0xb, 0xc, 0xd, 0xe, 0xf)  
  35.     .nelts = 5  
  36.     .next = 0x0   
  37. -------------------------------- the pool at the end:  
  38. -------------------------------- pool = 0x9208020 .d .last = 0x92080b8  
  39.     .end = 0x9208420  
  40.     .next = 0x0  
  41.     .failed = 0 .max = 984  
  42.   .current = 0x9208020  
  43.   .chain = 0x0  
  44.   .large = 0x0  
  45.   .cleanup = 0x0  
  46.   .log = 0x0 available pool memory = 872  

该例子中内存池和数组的(内存)物理结构可参考2.3节的图。

 

4. 小结

 

本文针对nginx-1.0.4的容器——链表结构进行了较为全面的分析,包括链表相关数据结构,链表创建和向链表中添加元素等。最后通过一个简单例子向读者展示nginx链表创建和添加元素操作,同时借此向读者展示编译测试代码的方法。

这篇关于nginx链表ngx_list_t的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/960230

相关文章

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

csu1329(双向链表)

题意:给n个盒子,编号为1到n,四个操作:1、将x盒子移到y的左边;2、将x盒子移到y的右边;3、交换x和y盒子的位置;4、将所有的盒子反过来放。 思路分析:用双向链表解决。每个操作的时间复杂度为O(1),用数组来模拟链表,下面的代码是参考刘老师的标程写的。 代码如下: #include<iostream>#include<algorithm>#include<stdio.h>#

深入手撕链表

链表 分类概念单链表增尾插头插插入 删尾删头删删除 查完整实现带头不带头 双向链表初始化增尾插头插插入 删查完整代码 数组 分类 #mermaid-svg-qKD178fTiiaYeKjl {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-

建立升序链表

题目1181:遍历链表 时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:2744 解决:1186 题目描述: 建立一个升序链表并遍历输出。 输入: 输入的每个案例中第一行包括1个整数:n(1<=n<=1000),接下来的一行包括n个整数。 输出: 可能有多组测试数据,对于每组数据, 将n个整数建立升序链表,之后遍历链表并输出。 样例输

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟)

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟) 题目描述 给定一个链表,链表中的每个节点代表一个整数。链表中的整数由 0 分隔开,表示不同的区间。链表的开始和结束节点的值都为 0。任务是将每两个相邻的 0 之间的所有节点合并成一个节点,新节点的值为原区间内所有节点值的和。合并后,需要移除所有的 0,并返回修改后的链表头节点。 思路分析 初始化:创建一个虚拟头节点

Collection List Set Map的区别和联系

Collection List Set Map的区别和联系 这些都代表了Java中的集合,这里主要从其元素是否有序,是否可重复来进行区别记忆,以便恰当地使用,当然还存在同步方面的差异,见上一篇相关文章。 有序否 允许元素重复否 Collection 否 是 List 是 是 Set AbstractSet 否

Windows下Nginx的安装及开机启动

1、将nginx-1.16.1.zip解压拷贝至D:\web\nginx目录下。 2、启动Nginx,两种方法: (1)直接双击nginx.exe,双击后一个黑色的弹窗一闪而过。 (2)打开cmd命令窗口,切换到nginx目录下,输入命令 nginx.exe 或者 start nginx ,回车即可。 3、检查nginx是否启动成功。 直接在浏览器地址栏输入网址 http://lo

nginx介绍及常用功能

什么是nginx nginx跟Apache一样,是一个web服务器(网站服务器),通过HTTP协议提供各种网络服务。 Apache:重量级的,不支持高并发的服务器。在Apache上运行数以万计的并发访问,会导致服务器消耗大量内存。操作系统对其进行进程或线程间的切换也消耗了大量的CPU资源,导致HTTP请求的平均响应速度降低。这些都决定了Apache不可能成为高性能WEB服务器  nginx:

学习记录:js算法(二十八):删除排序链表中的重复元素、删除排序链表中的重复元素II

文章目录 删除排序链表中的重复元素我的思路解法一:循环解法二:递归 网上思路 删除排序链表中的重复元素 II我的思路网上思路 总结 删除排序链表中的重复元素 给定一个已排序的链表的头 head , 删除所有重复的元素,使每个元素只出现一次 。返回 已排序的链表 。 图一 图二 示例 1:(图一)输入:head = [1,1,2]输出:[1,2]示例 2:(图

web群集--nginx配置文件location匹配符的优先级顺序详解及验证

文章目录 前言优先级顺序优先级顺序(详解)1. 精确匹配(Exact Match)2. 正则表达式匹配(Regex Match)3. 前缀匹配(Prefix Match) 匹配规则的综合应用验证优先级 前言 location的作用 在 NGINX 中,location 指令用于定义如何处理特定的请求 URI。由于网站往往需要不同的处理方式来适应各种请求,NGINX 提供了多种匹