nginx内存池源码学习及代码移植
通过学习nginx内存池源码了解其所创建的内存分配机制,并利用C++面向对象的思想将其封装为一个内存池类进行代码移植
为什么需要内存池
C/C++中通过malloc或new 的内存分配的主要缺点有:
一是可能需要花费很多时间,每次malloc或new 都要进入到内核,是一个效率相对较低的操作。
二是每次开辟的内存空间大小具有随机性,系统new操作是找到最近的一个足够大的连续空间分配出去,如果某个较小的内存被delete回收到系统,而之后发生的new操作需要更大尺寸的内存,于是较小的内存被弃用成为所谓的内存碎片。这样较小的内存被弃用在内存上就会将形成碎片,所以一个应用程序的运行会越来越慢。
内存池的出现就是为了提高向系统申请内存时的分配效率并减少内存碎片的出现(或对申请的内存进行充分的利用),内存池的主要思想是程序在申请堆上的内存时,我们可以给它一大块内存(远超出程序要申请的大小,即内存池),这样减少系统调用的次数,然后依据其所设计的逻辑对这块内存进行管理进行充分利用。
nginx内存池的数据结构及主要函数:
数据结构
//开辟的内存池的主体结构
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;//清理函数的handler的入口指针
ngx_log_t *log;
};
//小块内存数据头信息
typedef struct {
u_char *last;//可分配内存的开始位置
u_char *end;//可分配内存的末尾位置
ngx_pool_t *next;//指向下一个内存池的地址
ngx_uint_t failed;//记录当前内存吃分配失败的次