nginx内存池的主体结构:
ngx_pool_s
ngx_pool_data_t
ngx_pool_large_t
ngx_pool_cleanup_s
由 ngx_pool_data_x和ngx_pool_s构成:
ngx_pool_data_t:
typedef struct {
u_char *last;
u_char *end;
ngx_pool_t *next;
ngx_uint_t failed;
}ngx_pool_data_t;
last:unsigned char类型的指针,保存当前内存池分配到末位地址,下一次分配从此处开始。
end:内存池结束位置。
next:内存池里有很多块内存,这些内存块通过指针连成链表,next指向下一块内存。
failed:内存池分配失败次数,默认为四个成员
ngx_pool_s:
struct ngx_pool_s {
ngx_pool_data_t d;
size_t max;
ngx_pool_t *current;
ngx_chain_t *chain;
ngx_pool_large_t *large;
ngx_pool_cleanup_t *cleanup;
ngx_log_t *log;
};
d:内存池的数据块。
max:内存池数据块的最大值。
current:指向当前内存池。
chain:该指针挂接一个ngx_chain_t结构。
large:大块内存链表,即分配空间超过max的情况使用。
cleanup:内存池清理回调函数。
log:日志信息。
ngx_pool_large_s:
struct ngx_pool_large_s {
ngx_pool_large_t *next;
void *alloc;
};
next:指向下一个结点地址或NULL。
alloc:指向堆上申请的大块内存的起始地址或NULL。
内存池的基本操作:
创建内存池: 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);
1.内存池创建:nginx对内存的管理分为大内存与小内存,当某一个申请的内存大于某一个值时,就需要从大内存中分配空间,否则从小内存中分配空间。
nginx中的内存池是在创建的时候就设定好了大小,在以后分配小块内存的时候,如果内存不够,则是重新创建一块内存串到内存池中,而不是将原有的内存池进行扩张。当要分配大块内存是,则是在内存池外面再分配空间进行管理的,称为大块内存池。
2.内存申请:
ngx_align_ptr,这是一个用来内存地址取整的宏,非常精巧,一句话就搞定了。作用不言而喻,取整可以降低CPU读取内存的次数,提高性能。
ngx_palloc_block(ngx_pool_t *pool, size_t size),这个函数是用来分配新的内存块,为pool内存池开辟一个新的内存块,并申请使用size大小的内存;
ngx_palloc_large(ngx_pool_t *pool, size_t size),在ngx_palloc中首先会判断申请的内存大小是否超过内存块的最大限值,如果超过,则直接调用ngx_palloc_large,进入大内存块的分配流程;
ngx_pfree,我们在使用内存池时,可以使用ngx_palloc进行分配,使用ngx_pfree释放。因为大内存块的分配只对前3个内存块进行检查,否则就直接分配内存,所以大内存块的释放必须及时。
ngx_pool_cleanup_s,Nginx内存池支持通过回调函数,对外部资源的清理.ngx_pool_cleanup_t是回调函数结构体,它在内存池中以链表形式保存,在内存池进行销毁时,循环调用这些回调函数对数据进行清理。
ngx_destroy_pool这个函数用于销毁一个内存池;
为什么就遍历三个呢?
大块内存可以释放,为了提高分配效率,只遍历前三个。所以,大块内存要及时释放
为什么要先释放大块内存?
因为大块内存的结点信息是通过ngx_palloc_small申请出来的,结点信息都在内存块中,要是直接通过指针偏移,先释放小块内存,那大块内存结点信息就没了,就会造成内存泄漏,而且是大量内存的泄漏。
怎么销毁整个内存池?
摧毁内存池分三步
销毁内存池中数据
销毁大块内存
销毁小块内存
通过ngx_pool_cleanup_结构体中的handler函数指针销毁结构体中*data 指向的数据
销毁大块内存
销毁小块内存和free相同