刚开始分配的时候,内存池里面只有一个内存块,这块内存块总共拥有unit_count个单元。
如果这块内存块不够用,那么就要再搞一个新的内存块,新增的这块内存块里一共有grow_count个单元。
新增的这个内存块要和原来的内存块用链表连起来。相当于在链表的头部插入。memory_block指针就指向了链表里的第一块内存块。
unit_size表示了一个单元拥有几个字节。
class MemoryPool {
public:
int unit_count;
int grow_count;
int unit_size;
MemoryBlock* memory_block;}
nsize表示这个内存块里有多少个字节。就是单元数*每个单元的字节数得到的。
nfree表示这个内存块里有几个单元是空闲的。
nfirst是第一个空闲的单元,它的编号。编号就是这个内存块里第几个单元。
pNext指针指向下一个内存块。
address的设计非常巧妙。当申请一块内存块的时候,是申请了头部+身体这么多的内存。
头部里就包含了这个对象的成员变量,大小是sizeof(MemoryBlock)这个类的大小。
身体开始才是真正存放数据的地方。身体的大小就是nsize咯。
脖子的地方就是身体开始地方。address就是这个脖子,我们要用的不是address这个地址里存放的东西(*address),而是address这个地址。可以直接通过address来获取哦。
class MemoryBlock {
public:
int nsize;
int nfree;
int nfirst;
MemoryBlock* pNext;
char address[1];}
每个单元的前两个字节都存放了下一个空闲单元的编号。
所以内存块初始化的时候,要让这两个字节等于当前编号+1。
下面是内存块初始化的代码
MemoryBlock(int unitcount,int unitsize) {
nsize = unitcount * unitsize;
nfree = unitcount;
nfirst = 0;
pNext = NULL;
char *p = address;
for (int i = 0; i < unitcount - 1; i++) {
*(short*)p = i + 1;
p += unitsize;
}
*(short*)p = -1;
}
如果内存池里面还没有内存块,那肯定是先分配内存,调用内存块的构造函数,先搞一块内存块咯。
if (memory_block == NULL) {
memory_block = (MemoryBlock*)new(unit_count*(sizeof(MemoryBlock) + unit_size))MemoryBlock(unit_count, unit_size);
assert(memory_block != NULL);
}
有了内存块以后检查这个内存块还有没有空闲的单元可以使用的,没有的话去下一块内存块看看。
MemoryBlock *p = memory_block;
while (p != NULL and p->nfree == 0) {
p = p->pNext;
}
找了半天都没找到空闲的怎么办呢?只能再申请一块内存块了,还要把它插到内存块链表的头部。这里用头插法。
if (p == NULL) {
p = (MemoryBlock*)new(grow_count*(sizeof(MemoryBlock) + unit_size))MemoryBlock(grow_count, unit_size);
assert(p != NULL);
p->pNext = memory_block;
memory_block = p;
}
OK,现在找到这个内存块了。
内存块里的nfirst记录了第一块空闲的单元。
我们直接返回它不就行了。
可是这样的话,下次再想从这块内存块里面搞一个单元怎么办呢?
所以还是要做一下善后的工作,把nfirst指向下一块空闲的单元。
并且这个内存块的空闲单元数减一。
char* free_address=p->address + p->nfirst*p->nsize;
p->nfirst=*(short*)free_address;
p->nfree--;
return free_address;
既然你要释放,你首先要把你要释放的这个指针传给我吧。
我直接释放行不行呢?
肯定不行咯。
这里的释放不是说让你真的把这个内存单元给free掉。而是把它搞成空闲的状态。
所以首先找到这个单元所在的内存块。
MemoryBlock* pb = memory_block;
while (pb != NULL and p < pb->address||p>pb->address+pb->nsize) {
pb = pb->pNext;
}
要是没找到的话,这个单元就不是内存池分配的。
找到了以后呢。
先算出这个单元在这个内存块里编号n.
然后还是头插法。
把这个单元插到这块内存块第一个空闲单元的前面。
具体而言,就是让这个单元的前两个字节存放第一个空闲单元的编号。
然后让第一个空闲单元的编号变成这个单元的编号。
空闲单元的个数也可以加上去了。
if (pb != NULL) {
int n = (int)((char*)p - pb->address) / unit_size;
*(short*)p = pb->nfirst;
pb->nfirst = n;
pb->nfree++;
}
MemoryPool pool(3, 3, 4);
int *p1=(int*)pool.Alloc();
*p1 = 111;
pool.Free(p1);