我们知道C++中申请内存使用new, 释放内存使用delete.
char* p = new char[100]; // do something if (ch) { delete p; p = nullptr; }
这里有一个问题: new的时候指定了大小,为什么delete时不需要?
针对这个问题,我们需要了解new做了哪些事情. 正常情问下 new(replacement new除外)会做两件事情:
1. 调用malloc申请内存空间
2. 调用构造函数.
现在重点讲解一下 1. 调用malloc申请内存空间. 每次malloc申请的空间是大于用户的申请数量,见下图:
其中:
1. Header 为8字节,用于保存这片内存空间的大小(4G, 即32bit)及是否可用1bit. 考虑到内存对齐,占用8字节.保留上下两个header是为了方便内存回收,合并.
2. Data 为用户实际申请的大小.
3. Pad是用于补齐内存对齐的大小.
4. 如果在debug下,还会夹带debug信息.
通过上图,我们就可以理解为什么delete的时候不需要指定内存大小了.但同时也带出另外一个问题,假设我们每次都只是申请8个字节,系统给用户的大小是 8+8*2(两个header)+0(字节对齐)=24字节,比实际申请的大小多很多,如果申请一百万次,带来的额外开销就很大.
为了解决这个问题,又引入了内存池的概念.