首页 后端开发 php教程 Nginx内存管理

Nginx内存管理

Jul 30, 2016 pm 01:31 PM
log nbsp pool size

1.源代码位置

头文件:http://trac.nginx.org/nginx/browser/nginx/src/core/ngx_palloc.h

源文件:http://trac.nginx.org/nginx/browser/nginx/src/core/ngx_palloc.c

2.数据结构定义

先来学习一下nginx内存池的几个主要数据结构:

    ngx_pool_data_t(内存池数据块结构)

1:typedef struct { 2:     u_char               *last;        3:     u_char               *end; 4:     ngx_pool_t           *next; 5:     ngx_uint_t            failed; 6: } ngx_pool_data_t;
  • last:是一个unsigned char 类型的指针,保存的是/当前内存池分配到末位地址,即下一次分配从此处开始。
  • end:内存池结束位置;
  • next:内存池里面有很多块内存,这些内存块就是通过该指针连成链表的,next指向下一块内存。
  • failed:内存池分配失败次数。

ngx_pool_s(内存池头部结构)

1:struct ngx_pool_s { 2:     ngx_pool_data_t       d; 3:     size_t                max; 4:     ngx_pool_t           *current; 5:     ngx_chain_t          *chain; 6:     ngx_pool_large_t     *large; 7:     ngx_pool_cleanup_t   *cleanup; 8:     ngx_log_t            *log; 9: };
  • d:内存池的数据块;
  • max:内存池数据块的最大值;
  • current:指向当前内存池;
  • chain:该指针挂接一个ngx_chain_t结构;
  • large:大块内存链表,即分配空间超过max的情况使用;
  • cleanup:释放内存池的callback
  • log:日志信息

由ngx_pool_data_t和ngx_pool_t组成的nginx内存池结构如下图所示:

Nginx内存管理

3.相关函数介绍

在分析内存池方法前,需要对几个主要的内存相关函数作一下介绍:

ngx_alloc:(只是对malloc进行了简单的封装)

1:void * 2: ngx_alloc(size_t size, ngx_log_t *log) 3: { 4:     void  *p; 5:  6:     p = malloc(size); 7:     if (p == NULL) { 8:         ngx_log_error(NGX_LOG_EMERG, log, ngx_errno, 9:                       "malloc(%uz) failed", size); 10:     } 11:  12:     ngx_log_debug2(NGX_LOG_DEBUG_ALLOC, log, 0, "malloc: %p:%uz", p, size); 13:  14:     return p; 15: }

ngx_calloc:(调用malloc并初始化为0)

1:void * 2: ngx_calloc(size_t size, ngx_log_t *log) 3: { 4:     void  *p; 5:  6:     p = ngx_alloc(size, log); 7:  8:     if (p) { 9:         ngx_memzero(p, size); 10:     } 11:  12:     return p; 13: }

ngx_memzero:

1: #define ngx_memzero(buf, n)       (void) memset(buf, 0, n)

ngx_free :

1: #define ngx_free          free

ngx_memalign:

1:void * 2: ngx_memalign(size_t alignment, size_t size, ngx_log_t *log) 3: { 4:     void  *p; 5:     int    err; 6:  7:     err = posix_memalign(&p, alignment, size); 8:  9:     if (err) { 10:         ngx_log_error(NGX_LOG_EMERG, log, err, 11:                       "posix_memalign(%uz, %uz) failed", alignment, size); 12:         p = NULL; 13:     } 14:  15:     ngx_log_debug3(NGX_LOG_DEBUG_ALLOC, log, 0, 16:                    "posix_memalign: %p:%uz @%uz", p, size, alignment); 17:  18:     return p; 19: }

这里alignment主要是针对部分unix平台需要动态的对齐,对POSIX 1003.1d提供的posix_memalign( )进行封装,在大多数情况下,编译器和C库透明地帮你处理对齐问题。nginx中通过宏NGX_HAVE_POSIX_MEMALIGN来控制;调用posix_memalign( )成功时会返回size字节的动态内存,并且这块内存的地址是alignment的倍数。参数alignment必须是2的幂,还是void指针的大小的倍数。返回的内存块的地址放在了memptr里面,函数返回值是0.
 

4.内存池基本操作

  • 内存池对外的主要方法有:
创建内存池 ngx_pool_t *  ngx_create_pool(size_t size, ngx_log_t *log);
销毁内存池 void ngx_destroy_pool(ngx_pool_t *pool);
重置内存池 void ngx_reset_pool(ngx_pool_t *pool);
内存申请(对齐) void *  ngx_palloc(ngx_pool_t *pool, size_t size);
内存申请(不对齐) void *  ngx_pnalloc(ngx_pool_t *pool, size_t size);
内存清除 ngx_int_t  ngx_pfree(ngx_pool_t *pool, void *p);

4.1 创建内存池ngx_create_pool

ngx_create_pool用于创建一个内存池,我们创建时,传入我们的需要的初始大小:

1: ngx_pool_t * 2: ngx_create_pool(size_t size, ngx_log_t *log) 3: { 4:     ngx_pool_t  *p; 5:     6:     //以16(NGX_POOL_ALIGNMENT)字节对齐分配size内存 7:     p = ngx_memalign(NGX_POOL_ALIGNMENT, size, log); 8:     if (p == NULL) { 9:         return NULL; 10:     } 11:  12:     //初始状态:last指向ngx_pool_t结构体之后数据取起始位置 13:     p->d.last = (u_char *) p + sizeof(ngx_pool_t); 14:     //end指向分配的整个size大小的内存的末尾 15:     p->d.end = (u_char *) p + size; 16:     17:     p->d.next = NULL; 18:     p->d.failed = 0; 19:  20:     size = size - sizeof(ngx_pool_t); 21:     //#define NGX_MAX_ALLOC_FROM_POOL  (ngx_pagesize - 1),内存池最大不超过4095,x86中页的大小为4K 22:     p->max = (size 23:  24:     p->current = p; 25:     p->chain = NULL; 26:     p->large = NULL; 27:     p->cleanup = NULL; 28:     p->log = log; 29:  30:     return p; 31: }

nginx对内存的管理分为大内存与小内存,当某一个申请的内存大于某一个值时,就需要从大内存中分配空间,否则从小内存中分配空间。

nginx中的内存池是在创建的时候就设定好了大小,在以后分配小块内存的时候,如果内存不够,则是重新创建一块内存串到内存池中,而不是将原有的内存池进行扩张。当要分配大块内存是,则是在内存池外面再分配空间进行管理的,称为大块内存池。

 

4.2 内存申请 ngx_palloc

1:void * 2: ngx_palloc(ngx_pool_t *pool, size_t size) 3: { 4:     u_char      *m; 5:     ngx_pool_t  *p; 6:  7:     //如果申请的内存大小小于内存池的max值 8:     if (size max) { 9:  10:         p = pool->current; 11:  12:         do { 13:             //对内存地址进行对齐处理 14:             m = ngx_align_ptr(p->d.last, NGX_ALIGNMENT); 15:  16:             //如果当前内存块够分配内存,则直接分配 17:             if ((size_t) (p->d.end - m) >= size) 18:             { 19:                 p->d.last = m + size; 20:  21:                 return m; 22:             } 23:             24:             //如果当前内存块有效容量不够分配,则移动到下一个内存块进行分配 25:             p = p->d.next; 26:  27:         } while (p); 28:  29:         //当前所有内存块都没有空闲了,开辟一块新的内存,如下2详细解释 30:         return ngx_palloc_block(pool, size); 31:     } 32:  33:     //分配大块内存 34:     return ngx_palloc_large(pool, size); 35: }

需要说明的几点:

1、ngx_align_ptr,这是一个用来内存地址取整的宏,非常精巧,一句话就搞定了。作用不言而喻,取整可以降低CPU读取内存的次数,提高性能。因为这里并没有真正意义调用malloc等函数申请内存,而是移动指针标记而已,所以内存对齐的活,C编译器帮不了你了,得自己动手。

1: #define ngx_align_ptr(p, a)                                                   \ 2:      (u_char *) (((uintptr_t) (p) + ((uintptr_t) a - 1)) & ~((uintptr_t) a - 1))

2、开辟一个新的内存块 ngx_palloc_block(ngx_pool_t *pool, size_t size)

这个函数是用来分配新的内存块,为pool内存池开辟一个新的内存块,并申请使用size大小的内存;

1:static void * 2: ngx_palloc_block(ngx_pool_t *pool, size_t size) 3: { 4:     u_char      *m; 5:     size_t       psize; 6:     ngx_pool_t  *p, *new; 7:  8:     //计算内存池第一个内存块的大小 9:     psize = (size_t) (pool->d.end - (u_char *) pool); 10:  11:     //分配和第一个内存块同样大小的内存块 12:     m = ngx_memalign(NGX_POOL_ALIGNMENT, psize, pool->log); 13:     if (m == NULL) { 14:         return NULL; 15:     } 16:  17:     new = (ngx_pool_t *) m; 18:  19:     //设置新内存块的end 20:     new->d.end = m + psize; 21:     new->d.next = NULL; 22:     new->d.failed = 0; 23:  24:     //将指针m移动到d后面的一个位置,作为起始位置 25:     m += sizeof(ngx_pool_data_t); 26:     //对m指针按4字节对齐处理 27:     m = ngx_align_ptr(m, NGX_ALIGNMENT); 28:     //设置新内存块的last,即申请使用size大小的内存 29:     new->d.last = m + size; 30:  31:     //这里的循环用来找最后一个链表节点,这里failed用来控制循环的长度,如果分配失败次数达到5次,就忽略,不需要每次都从头找起 32:     for (p = pool->current; p->d.next; p = p->d.next) { 33:         if (p->d.failed++ > 4) { 34:             pool->current = p->d.next; 35:         } 36:     } 37:  38:     p->d.next = new; 39:  40:     return m; 41: }

3、分配大块内存 ngx_palloc_large(ngx_pool_t *pool, size_t size)

在ngx_palloc中首先会判断申请的内存大小是否超过内存块的最大限值,如果超过,则直接调用ngx_palloc_large,进入大内存块的分配流程;

1:static void * 2: ngx_palloc_large(ngx_pool_t *pool, size_t size) 3: { 4:     void              *p; 5:     ngx_uint_t         n; 6:     ngx_pool_large_t  *large; 7:  8:     // 直接在系统堆中分配一块大小为size的空间 9:     p = ngx_alloc(size, pool->log); 10:     if (p == NULL) { 11:         return NULL; 12:     } 13:  14:     n = 0; 15:  16:     // 查找到一个空的large区,如果有,则将刚才分配的空间交由它管理  17:     for (large = pool->large; large; large = large->next) { 18:         if (large->alloc == NULL) { 19:             large->alloc = p; 20:             return p; 21:         } 22:         //为了提高效率, 如果在三次内没有找到空的large结构体,则创建一个 23:         if (n++ > 3) { 24:             break; 25:         } 26:     } 27:  28:  29:     large = ngx_palloc(pool, sizeof(ngx_pool_large_t)); 30:     if (large == NULL) { 31:         ngx_free(p); 32:         return NULL; 33:     } 34:     35:     //将large链接到内存池 36:     large->alloc = p; 37:     large->next = pool->large; 38:     pool->large = large; 39:  40:     return p; 41: }

整个内存池分配如下图:

Nginx内存管理

  • 4.3 内存池重置 ngx_reset_pool

1:void 2: ngx_reset_pool(ngx_pool_t *pool) 3: { 4:     ngx_pool_t        *p; 5:     ngx_pool_large_t  *l; 6:     7:     //释放大块内存 8:     for (l = pool->large; l; l = l->next) { 9:         if (l->alloc) { 10:             ngx_free(l->alloc); 11:         } 12:     } 13:     14:     // 重置所有小块内存区 15:     for (p = pool; p; p = p->d.next) { 16:         p->d.last = (u_char *) p + sizeof(ngx_pool_t); 17:         p->d.failed = 0; 18:     } 19:  20:     pool->current = pool; 21:     pool->chain = NULL; 22:     pool->large = NULL; 23: }

4.4 内存池释放 ngx_pfree

1: ngx_int_t 2: ngx_pfree(ngx_pool_t *pool, void *p) 3: { 4:     ngx_pool_large_t  *l; 5:  6:     //只检查是否是大内存块,如果是大内存块则释放 7:     for (l = pool->large; l; l = l->next) { 8:         if (p == l->alloc) { 9:             ngx_log_debug1(NGX_LOG_DEBUG_ALLOC, pool->log, 0, 10:                            "free: %p", l->alloc); 11:             ngx_free(l->alloc); 12:             l->alloc = NULL; 13:  14:             return NGX_OK; 15:         } 16:     } 17: 

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

热门话题

Java教程
1653
14
CakePHP 教程
1413
52
Laravel 教程
1305
25
PHP教程
1251
29
C# 教程
1224
24
解决方法:您的组织要求您更改 PIN 码 解决方法:您的组织要求您更改 PIN 码 Oct 04, 2023 pm 05:45 PM

“你的组织要求你更改PIN消息”将显示在登录屏幕上。当在使用基于组织的帐户设置的电脑上达到PIN过期限制时,就会发生这种情况,在该电脑上,他们可以控制个人设备。但是,如果您使用个人帐户设置了Windows,则理想情况下不应显示错误消息。虽然情况并非总是如此。大多数遇到错误的用户使用个人帐户报告。为什么我的组织要求我在Windows11上更改我的PIN?可能是您的帐户与组织相关联,您的主要方法应该是验证这一点。联系域管理员会有所帮助!此外,配置错误的本地策略设置或不正确的注册表项也可能导致错误。即

Windows 11 上调整窗口边框设置的方法:更改颜色和大小 Windows 11 上调整窗口边框设置的方法:更改颜色和大小 Sep 22, 2023 am 11:37 AM

Windows11将清新优雅的设计带到了最前沿;现代界面允许您个性化和更改最精细的细节,例如窗口边框。在本指南中,我们将讨论分步说明,以帮助您在Windows操作系统中创建反映您的风格的环境。如何更改窗口边框设置?按+打开“设置”应用。WindowsI转到个性化,然后单击颜色设置。颜色更改窗口边框设置窗口11“宽度=”643“高度=”500“>找到在标题栏和窗口边框上显示强调色选项,然后切换它旁边的开关。若要在“开始”菜单和任务栏上显示主题色,请打开“在开始”菜单和任务栏上显示主题

如何在 Windows 11 上更改标题栏颜色? 如何在 Windows 11 上更改标题栏颜色? Sep 14, 2023 pm 03:33 PM

默认情况下,Windows11上的标题栏颜色取决于您选择的深色/浅色主题。但是,您可以将其更改为所需的任何颜色。在本指南中,我们将讨论三种方法的分步说明,以更改它并个性化您的桌面体验,使其具有视觉吸引力。是否可以更改活动和非活动窗口的标题栏颜色?是的,您可以使用“设置”应用更改活动窗口的标题栏颜色,也可以使用注册表编辑器更改非活动窗口的标题栏颜色。若要了解这些步骤,请转到下一部分。如何在Windows11中更改标题栏的颜色?1.使用“设置”应用按+打开设置窗口。WindowsI前往“个性化”,然

Windows 11 上启用或禁用任务栏缩略图预览的方法 Windows 11 上启用或禁用任务栏缩略图预览的方法 Sep 15, 2023 pm 03:57 PM

任务栏缩略图可能很有趣,但它们也可能分散注意力或烦人。考虑到您将鼠标悬停在该区域的频率,您可能无意中关闭了重要窗口几次。另一个缺点是它使用更多的系统资源,因此,如果您一直在寻找一种提高资源效率的方法,我们将向您展示如何禁用它。不过,如果您的硬件规格可以处理它并且您喜欢预览版,则可以启用它。如何在Windows11中启用任务栏缩略图预览?1.使用“设置”应用点击键并单击设置。Windows单击系统,然后选择关于。点击高级系统设置。导航到“高级”选项卡,然后选择“性能”下的“设置”。在“视觉效果”选

OOBELANGUAGE错误Windows 11 / 10修复中出现问题的问题 OOBELANGUAGE错误Windows 11 / 10修复中出现问题的问题 Jul 16, 2023 pm 03:29 PM

您是否在Windows安装程序页面上看到“出现问题”以及“OOBELANGUAGE”语句?Windows的安装有时会因此类错误而停止。OOBE表示开箱即用的体验。正如错误提示所表示的那样,这是与OOBE语言选择相关的问题。没有什么可担心的,你可以通过OOBE屏幕本身的漂亮注册表编辑来解决这个问题。快速修复–1.单击OOBE应用底部的“重试”按钮。这将继续进行该过程,而不会再打嗝。2.使用电源按钮强制关闭系统。系统重新启动后,OOBE应继续。3.断开系统与互联网的连接。在脱机模式下完成OOBE的所

Windows 11 上的显示缩放比例调整指南 Windows 11 上的显示缩放比例调整指南 Sep 19, 2023 pm 06:45 PM

在Windows11上的显示缩放方面,我们都有不同的偏好。有些人喜欢大图标,有些人喜欢小图标。但是,我们都同意拥有正确的缩放比例很重要。字体缩放不良或图像过度缩放可能是工作时真正的生产力杀手,因此您需要知道如何对其进行自定义以充分利用系统功能。自定义缩放的优点:对于难以阅读屏幕上的文本的人来说,这是一个有用的功能。它可以帮助您一次在屏幕上查看更多内容。您可以创建仅适用于某些监视器和应用程序的自定义扩展配置文件。可以帮助提高低端硬件的性能。它使您可以更好地控制屏幕上的内容。如何在Windows11

10种在 Windows 11 上调整亮度的方法 10种在 Windows 11 上调整亮度的方法 Dec 18, 2023 pm 02:21 PM

屏幕亮度是使用现代计算设备不可或缺的一部分,尤其是当您长时间注视屏幕时。它可以帮助您减轻眼睛疲劳,提高易读性,并轻松有效地查看内容。但是,根据您的设置,有时很难管理亮度,尤其是在具有新UI更改的Windows11上。如果您在调整亮度时遇到问题,以下是在Windows11上管理亮度的所有方法。如何在Windows11上更改亮度[10种方式解释]单显示器用户可以使用以下方法在Windows11上调整亮度。这包括使用单个显示器的台式机系统以及笔记本电脑。让我们开始吧。方法1:使用操作中心操作中心是访问

如何修复Windows服务器中的激活错误代码0xc004f069 如何修复Windows服务器中的激活错误代码0xc004f069 Jul 22, 2023 am 09:49 AM

Windows上的激活过程有时会突然转向显示包含此错误代码0xc004f069的错误消息。虽然激活过程已经联机,但一些运行WindowsServer的旧系统可能会遇到此问题。通过这些初步检查,如果这些检查不能帮助您激活系统,请跳转到主要解决方案以解决问题。解决方法–关闭错误消息和激活窗口。然后,重新启动计算机。再次从头开始重试Windows激活过程。修复1–从终端激活从cmd终端激活WindowsServerEdition系统。阶段–1检查Windows服务器版本您必须检查您使用的是哪种类型的W

See all articles