STL源码剖析(空间配置器)

Posted Runnyu

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了STL源码剖析(空间配置器)相关的知识,希望对你有一定的参考价值。

前言

在STL中,容器的定义中都带一个模板参数,如vector

template <class T, class Alloc = alloc>
class vector {...}

其中第二个参数就是该容器使用的空间配置器,其中缺省使用STL已经实现的空间配置器(alloc),

该配置器使用malloc/free等为vector分配内存。

 

缺省的空间配置器

alloc定义了两级的空间配置器,第一级是对malloc/free简单的封装。

而为了解决内存碎片的问题,跟进行内存管理,alloc实现的第二级的空间配置器。

第二级空间配置器在分配大块内存(大于128bytes)时,会直接调用第一级空间配置器,

而分配小于128bytes的内存时,则使用内存池跟free_list进行内存分配/管理。

 

第一级空间配置器

基本实现如下(跟SGI STL可能有点出入,主要是提取核心的内容)

 1 class base_alloc {
 2 public:
 3     // 只是对malloc/free的简单封装
 4     static void* allocate(size_t n)
 5     {
 6         void* res = malloc(n);
 7         if (0 == res) res = oom_malloc(n);
 8         return res;
 9     }
10     static void* reallocate(void* p, size_t new_sz)
11     {
12         void* res = realloc(p, new_sz);
13         if (0 == res) res = oom_realloc(p, new_sz);
14         return res;
15     }
16     static void deallocate(void* p)
17     {
18         free(p);
19     }
20     // 用来设置内存不足时的处理函数 该函数参数跟返回值都是一个函数指针
21     // 一般会抛出异常/尝试回收内存
22     static void(*set_handler(void(*f)()))()
23     {
24         void(*old)() = _oom_handler;
25         _oom_handler = f;
26         return old;
27     }
28 private:
29     // 用来处理内存不足的情况
30     static void* oom_malloc(size_t n)
31     {
32         void(*my_handler)();
33         void* res;
34 
35         for (;;)
36         {
37             my_handler = _oom_handler;
38             if (0 == my_handler) { return NULL; }
39             (*my_handler)();
40             if (res = malloc(n)) return res;
41         }
42     }
43     // 用来处理内存不足的情况
44     static void* oom_realloc(void* p, size_t n)
45     {
46         void(*my_handler)();
47         void* res;
48 
49         for (;;)
50         {
51             my_handler = _oom_handler;
52             if (0 == my_handler) { return NULL; }
53             (*my_handler)();
54             if (res = reallocate(p, n)) return res;
55         }
56     }
57     // 由用户设置,在内存不足的时候进行处理,由上面两个函数调用
58     static void(*_oom_handler)();
59 };
60 
61 // 处理函数默认为0 
62 void(*base_alloc::_oom_handler)() = 0;
View Code

它可以设定一个处理内存不足的时候的处理函数(跟set_new_handler类似)。

 

第二级空间配置器

该配置器维护一个free_list,这是一个指针数组。

在分配内存的时候,补足8bytes的倍数,free_list数组中每个指针分别管理分配大小为8、16、24、32...bytes的内存。

下图表示从二级空间配置器中分配内存时是如何维护free_list的(建议参考下面源码阅读)。

开始所有指针都为0,没有可分配的区块时(就是free_list[i]==0)会从内存池中分配内存(默认分配20个区块)插入到free_list[i]中。

然后改变free_list[i]的指向,指向下一个区块(free_list_link指向下一个区块,如果没有则为0)。

 

下图表示二级空间配置器回收内存时是如何维护free_list结构的。

回收的时候只是将区块插入到free_list[i]的开头,这块内存用于下次分配的时候使用。

该配置器实现如下(同样提取核心的部分):

  1 enum { _ALIGN = 8 };    // 对齐
  2 enum { _MAX_BYTES = 128 }; // 区块大小上限
  3 enum { _NFREELISTS = _MAX_BYTES / _ALIGN }; // free-list个数
  4 
  5 class default_alloc {
  6 private:
  7     // 将bytes上调到8的倍数
  8     static size_t ROUND_UP(size_t bytes)
  9     {
 10         return (bytes + _ALIGN - 1) & ~(_ALIGN - 1);
 11     }
 12 private:
 13     union obj {
 14         union obj* free_list_link;
 15         char client_data[1];
 16     };
 17 private:
 18     // 16个free-lists 各自管理分别为8,16,24...的小额区块
 19     static obj* free_list[_NFREELISTS];
 20     // 根据区块大小,决定使用第n号free-list
 21     static size_t FREELIST_INDEX(size_t bytes)
 22     {
 23         return (bytes + _ALIGN - 1) / _ALIGN - 1;
 24     }
 25     // 分配内存,返回一个大小为n的区块,可能将大小为n的其他区块加入到free_list
 26     static void* refill(size_t n)
 27     {
 28         // 默认分配20个区块
 29         int nobjs = 20;
 30         char* chunk = chunk_alloc(n, nobjs);
 31         obj** my_free_list;
 32         obj* result, *current_obj, *next_obj;
 33 
 34         // 如果只分配了一个区块,直接返回
 35         if (1 == nobjs) return chunk;
 36         // 否则将其他区块插入到free list
 37         my_free_list = free_list + FREELIST_INDEX(n);
 38         result = (obj*)chunk;
 39         // 第一个区块返回 后面的区块插入到free list
 40         *my_free_list = next_obj = (obj*)(chunk + n);
 41         for (int i = 1;; ++i)
 42         {
 43             current_obj = next_obj;
 44             next_obj = (obj*)((char*)next_obj + n);
 45             // 最后一个next的free_list_link为0
 46             if (nobjs - 1 == i)
 47             {
 48                 current_obj->free_list_link = 0;
 49                 break;
 50             }
 51             current_obj->free_list_link = next_obj;
 52         }
 53         return result;
 54     }
 55     // 分配内存
 56     // 在内存池容量足够时,只调整start_free跟end_free指针
 57     // 在内存池容量不足时,调用malloc分配内存(2 * size * nobjs +  ROUND_UP(heap_size >> 4),每次调整heap_size += 本次分配内存的大小) 
 58     static char* chunk_alloc(size_t size, int& nobjs);
 59 
 60     static char* start_free; //内存池的起始位置
 61     static char* end_free;     //内存池的结束位置
 62     static size_t heap_size;         //分配内存时的附加量
 63 public:
 64     static void* allocate(size_t n)
 65     {
 66         obj** my_free_list;
 67         obj* result;
 68         // 大于128就调用第一级空间配置器
 69         if (n > (size_t)_MAX_BYTES)
 70             return base_alloc::allocate(n);
 71 
 72         // 寻找适当的free-list
 73         my_free_list = free_list + FREELIST_INDEX(n);
 74         result = *my_free_list;
 75         // 没有可用的free list
 76         if (result == 0)
 77             return refill(ROUND_UP(n));
 78 
 79         // 调整free list 移除free list
 80         *my_free_list = result->free_list_link;
 81         return result;
 82     }
 83     static void deallocate(void* p, size_t n)
 84     {
 85         obj* q = (obj*)p;
 86         obj** my_free_list;
 87 
 88         // 大于128就调用第一级空间配置器
 89         if (n > (size_t)_MAX_BYTES)
 90         {
 91             base_alloc::deallocate(p);
 92             return;
 93         }
 94 
 95         // 寻找对应的free list
 96         my_free_list = free_list + FREELIST_INDEX(n);
 97         // 调整free list 回收区块(将区块插入到my_free_list)
 98         q->free_list_link = *my_free_list;
 99         *my_free_list = q;
100     }
101     static void reallocate(void* p, size_t old_sz, size_t new_sz);
102 };
103 
104 char* default_alloc::start_free = 0;
105 char* default_alloc::end_free = 0;
106 size_t default_alloc::heap_size = 0;
107 default_alloc::obj* default_alloc::free_list[_NFREELISTS] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 };
View Code

 

以上是关于STL源码剖析(空间配置器)的主要内容,如果未能解决你的问题,请参考以下文章

STL源码剖析第二章---配置器

STL源码剖析——空间配置器Allocator#2 一/二级空间配置器

STL源码剖析 第二章 空间配置器

STL源码剖析之allocator

STL源码剖析——空间配置器Allocator#3 自由链表与内存池

《STL源码剖析》要点摘抄