自己动手实现STL 01:内存配置器的实现(stl_alloc.h)

一、前言

  在STL中,容器是其中的重中之重,基本的STL中的算法,仿函数等都是围绕着容器实现的功能。而,内存配置器,是容器的实现的基础。所以,我第一次要去编写便是内存配置器的实现。在STL中,内存配置器的实现是在stl_alloc.h中。

二、配置器原理简要介绍

  在SGI STL中配置分为两级,第一级配置器和第二级配置器。两者关系如下:

图1:第一级配置器和第二级配置器

  在SGI STL中内存的配置器分为两级,第一级配置器和第二级配置器。第一级配置器就是,直接调用系统的malloc分配内存。对于小于128byte的内存分配请求,我们使用第二级内存配置器。第二级内存配置器是一个内存池,其中共有16个已分配好的区块形成的链表。这16个链表的中区块的大小依次是8,16,24....128byte,都8的倍数。每次请求小于等于128byte时,把请求的大小上调到最接近的8的倍数,比如,7就上调为8,30就上调为32,然后找到对应大小区块的链表,从中取下一个区块返回给请求。

  第二级配置器使用内存池的好处就是,可以避免太多小额区块造成的内存破碎。同时,每次分配内存都需要调用malloc去分配,malloc调用的消耗的时间等资源是一定的,对于大区块的分配这样的消耗的时间等资源,是没有什么的。但是对于小额区块的,它的消耗就显得太不值的了。我们可以采用一次预分配一块连续的大区块,把它串成一个定额大小的区块链表,(8的倍数字节),下次使用的时候,从对应预分配的区块链表中找一个能够满足大小的,最接近的区块直接返回给请求,这样就可以避免对于小区块的malloc调用。同时对于小区块的释放,可以直接把它加入到内存池中对应大小的链表中即可。

图2:把连续大区块串成小额区块

  在第二级配置器中维持一个free_list[16]数组,其中存储着8-128byte各种大小区块的链表的头节点地址。

图3:free_list[16]结构

  每次分配只要从适当大小的链表中取出第一个节点,返回给请求,让free_list对应的位置的保存链表的下一个节点地址。释放的时候,对于小于等于128byte的区块,只要把它插入对应大小区块链表的头,然后调整保存的链表头结点的地址就可以了。当需求大小区块的使用完了的时候,可以利用malloc一次分配适当大小的大区块,然后把它分割为对应大小的小区块,在把它们串联起来形成链表,把第一个节点地址存入对应的free_list的位置中。

三、实现源代码

  上面只是配置器的简介,在源码中有详细的注释。源码如下:

  1 /*************************************************************************
  2     > File Name: stl_alloc_wjzh.h
  3     > Author: wjzh
  4     > Mail: [email protected]
  5     > Created Time: 2014年10月31日 星期五 16时06分23秒
  6  ************************************************************************/
  7
  8 #ifndef __SGI_STL_INTERNAL_ALLOC_WJZH_H
  9 #define __SGI_STL_INTERNAL_ALLOC_WJZH_H
 10
 11 #ifdef __SUNPRO_CC
 12 #    define __PRIVATE public
 13 #else
 14 #    define __PRIVATE private
 15 #endif
 16
 17 #ifdef __STL_STATIC_TEMPLATE_MEMBER_BUG
 18 #    define __USE_MALLOC
 19 #endif
 20
 21 #if 0
 22 #    include <new>
 23 #    define __THROW_BAD_ALLOC throw bad_alloc
 24 #elif !defined(__THROW_BAD_ALLOC)
 25 //#    include <iostream.h>
 26 #include <iostream>
 27 #    define __THROW_BAD_ALLOC std::cerr << "out of memory" << std::endl; exit(1)
 28 #endif
 29
 30 #ifndef __ALLOC
 31 #    define __ALLOC alloc
 32 #endif
 33 #ifdef __STL_WIN32THREADS
 34 #    include <windows.h>
 35 #endif
 36
 37 #include <stddef.h>
 38 #include <stdlib.h>
 39 #include <string.h>
 40 #include <assert.h>
 41 #ifndef __RESTRICT
 42 #    define __RESTRICT
 43 #endif
 44
 45 #if !defined(__STL_PTHREADS) && !defined(_NOTHREADS)  46  && !defined(__STL_SGI_THREADS) && !defined(__STL_WIN32THREADS)
 47 #    define _NOTHREADS
 48 #endif
 49
 50 #ifdef __STL_PTHREADS
 51 #    include <pthread.h>
 52 #    define __NODE_ALLOCATOR_LOCK  53         if (threads) pthread_mutex_lock(&__node_allocator_lock)
 54 #    define __NODE_ALLOCATOR_UNLOCK  55         if (threads) pthread_mutex_unlock(&__node_allocator_lock)
 56 #    define __NODE_ALLOCATOR_THREADS true
 57 #    define __VOLATILE volatile
 58 # endif
 59 # ifdef __STL_WIN32THREADS
 60 #    define __NODE_ALLOCATOR_LOCK  61         EnterCriticalSection(&__node_allocator_lock)
 62 #    define __NODE_ALLOCATOR_UNLOCK  63         LeaveCriticalSection(&__node_allocator_lock)
 64 #    define __NODE_ALLOCATOR_THREADS true
 65 #    define __VOLATILE volatile
 66 # endif
 67
 68 # ifdef __STL_SGI_THREADS
 69     extern "C" {
 70         extern int __us_rsthread_malloc;
 71     }
 72 #    define __NODE_ALLOCATOR_LOCK if (threads && __us_rsthread_malloc)  73 { __lock(&__node_allocator_lock);    }
 74 #    define __NODE_ALLOCATOR_UNLOCK if(threads && __us_rsthread_malloc)  75 { __unlcok(&__node_allocator_lock);    }
 76 #    define __NODE_ALLOCATOR_THREADS true
 77 #    define __VOLATILE volatile
 78 # endif
 79
 80 # ifdef _NOTHREADS
 81 #    define __NODE_ALLOCATOR_LOCK
 82 #    define __NODE_ALLOCATOR_UNLOCK
 83 #    define __NODE_ALLOCATOR_THREADS false
 84 #    define __VOLATILE
 85 # endif
 86
 87 __STL_BEGIN_NAMESPACE
 88
 89 #if defined(__sgi) && !defined(__GNUC__) && (_MIPS_SIM != _MIPS_SIM_ABI32)
 90 #pragma set woff 1174
 91 #endif
 92
 93 #ifdef __STL_STATIC_TEMPLATE_MEMBER_BUG
 94 #    ifdef __DECLARE_GLOBALS_HERE
 95         void (* __malloc_alloc_oom_handler)() = 0;
 96 #else
 97         extern void (* __malloc_alloc_oom_handler)();
 98 # endif
 99 #endif
100
101 template <int inst>
102 class __malloc_alloc_template {
103     private:
104         static void *oom_malloc(size_t);
105         static void *oom_realloc(void *, size_t);
106 #ifndef __STL_STATIC_TEMPLATE_MEMBER_BUG
107         static void (* __malloc_alloc_oom_handler)();
108 #endif
109     public:
110         static void* allocate(size_t n)
111         {
112             void *result = malloc(n);
113             if (0 == result) result = oom_malloc(n);
114             return result;
115         }
116
117         static void deallocate(void *p, size_t)
118         {
119             free(p);
120         }
121
122         static void * reallocate(void *p, size_t , size_t new_sz)
123         {
124             void *result = realloc(p, new_sz);
125             if (0 == result) result = oom_realloc(p, new_sz);
126             return result;
127         }
128
129         static void (* set_malloc_handler(void (*f)()))()
130         {
131             void (* old)() = __malloc_alloc_oom_handler;
132             __malloc_alloc_oom_handler = f;
133             return(old);
134         }
135
136 };
137
138 // malloc_alloc out-of-memory handling
139 // 分配内存时,没有内存时的处理
140
141 #ifndef __STL_STATIC_TEMPLATE_MEMBER_BUG
142 template <int inst>
143 void (* __malloc_alloc_template<inst>::__malloc_alloc_oom_handler)() = 0;
144 #endif
145
146 template <int inst>
147 void * __malloc_alloc_template<inst>::oom_malloc(size_t n)
148 {
149     void (* my_malloc_handler)();
150     void *result;
151     for (;;)
152     {
153         my_malloc_handler = __malloc_alloc_oom_handler;
154         if (0 == my_malloc_handler) { __THROW_BAD_ALLOC; }
155         (*my_malloc_handler)();
156         result = malloc(n);
157         if (result) return (result);
158     }
159 }
160
161 template <int inst>
162 void * __malloc_alloc_template<inst>::oom_realloc(void *p, size_t n)
163 {
164     void (* my_malloc_handler)();
165     void *result;
166
167     for (;;)
168     {
169         my_malloc_handler = __malloc_alloc_oom_handler;
170         if (0 == my_malloc_handler) { __THROW_BAD_ALLOC; }
171         (*my_malloc_handler)();
172         result = realloc(p, n);
173         if (result) return (result);
174     }
175 }
176
177 typedef __malloc_alloc_template<0> malloc_alloc;
178
179 template<class T, class Alloc>
180 class simple_alloc
181 {
182     public:
183         static T *allocate(size_t n)
184         {
185             return 0 == n ? 0 : (T*) Alloc::allocate(n * sizeof(T));
186         }
187         static T *allocate(void)
188         {
189             return (T*) Alloc::allocate(sizeof(T));
190         }
191         static void deallocate(T *p, size_t n)
192         {
193             if (0 != n) Alloc::deallocate(p, n * sizeof(T));
194         }
195         static void deallocate(T *p)
196         {
197             Alloc::deallocate(p, sizeof(T));
198         }
199 };
200
201 // Allocator adaptor to check size arguments for debugging.
202 template <class Alloc>
203 class debug_alloc
204 {
205     private:
206         enum { extra = 8};    // Size of space used to store size. Note
207                     // that this must be large enough to preserve
208                     // alignment.
209
210     public:
211         static void * allocate(size_t n)
212         {
213             char *result = (char *)Alloc::allocate(n + extra);
214             *(size_t *)result = n;    //前size_t大小用来记录result的大小,实际预分配了extra个字节,用来存储大小,
215                         //但是只用size_t字节,因为不同系统size_t大小不同,8个字节足够满足所有系统了
216             return result + extra;
217         }
218
219         static void deallocate(void *p, size_t n)
220         {
221             char * real_p = (char *)p - extra;
222             assert(*(size_t *)real_p == n);
223             Alloc::deallocate(real_p, n + extra);
224         }
225
226         static void * reallocate(void *p, size_t old_sz, size_t new_sz)
227         {
228             char * real_p = (char *)p - extra;
229             assert(*(size_t *)real_p == old_sz);
230             char * result = (char *)
231                             Alloc::reallocate(real_p, old_sz + extra, new_sz+ extra);
232             *(size_t *)result = new_sz;
233             return result + extra;
234         }
235
236 };
237
238 #ifdef __USE_MALLOC
239
240 typedef malloc_alloc alloc;
241 typedef malloc_alloc single_client_alloc;
242
243 #else
244
245 //下面是第二级配置器
246 //主要是维护一个内存池,用来小于128byte的小型区块内存的分配
247 //其中,有多个链表,各链表中的node大小从8-128byte,都是8的倍数
248 //分配时,不是8的倍数,上调至最近的8的倍数,
249 //然后从相应链表中取下一个对应大小的node分配给请求
250 #ifdef __SUNPRO_CC
251     enum {__ALIGN = 8};  //小型区块的上调边界
252     enum {__MAX_BYTES = 128};   //小型区块的上限
253     enum {__NFREELISTS = __MAX_BYTES/__ALIGN};
254 #endif
255
256 //第二级配置器
257 template <bool threads, int inst>
258 class __default_alloc_template
259 {
260     private:
261 # ifndef __SUNPRO_CC
262     enum {__ALIGN = 8};  //小型区块的上调边界
263     enum {__MAX_BYTES = 128};   //小型区块的上限
264     enum {__NFREELISTS = __MAX_BYTES/__ALIGN};
265 # endif
266     //大小上调至8的倍数
267     static size_t ROUND_UP(size_t bytes)
268     {
269         return (((bytes) + __ALIGN-1) & ~(__ALIGN - 1));
270     }
271 __PRIVATE:
272     union obj
273     {
274         union obj * free_list_link;  //用于在链表中指向下一个节点
275         char client_data[1]; //用于存储实际区块的内存地址,由于这是一个union,很好的节约了这个数据的内存
276     };
277     private:
278 # ifdef __SUNPRO_CC
279     static obj * __VOLATILE free_list[];
280 # else
281     static obj * __VOLATILE free_list[__NFREELISTS];
282 # endif
283     static size_t FREELIST_INDEX(size_t bytes)
284     {
285         return (((bytes) + __ALIGN-1)/__ALIGN - 1);
286     }
287
288     //返回大小为n的对象,并可能加入大小为n的其他区块到free list
289     static void *refill(size_t n);
290     //配置一块空间,可容纳nobjs个大小为"size"的区块
291     //如果配置nobjs个区块有所不便,nobjs可能会降低
292     static char *chunk_alloc(size_t size, int &nobjs);
293
294     //chunk 分配、配置的状态
295     static char *start_free; //内存池起始位置。只在chunk_alloc()中变化
296     static char *end_free;   //内存池结束位置。只在chunk_alloc()中变化
297     static size_t heap_size;
298 /*
299 # ifdef __STL_SGI_THREADS
300     static volatile unsigned long __node_allocator_lock;
301     static void __lock(volatile unsigned long *);
302     static inline void __unlock(volatile unsigned long *);
303 # endif
304 */
305
306 # ifdef __STL_PTHREADS
307     static pthread_mutex_t __node_allocator_lock;
308 # endif
309
310 # ifdef __STL_WIN32THREADS
311     static CRITICAL_SECTION __node_allocator_lock;
312     static bool __node_allocator_lock_initialized;
313
314     public:
315         __default_alloc_template()
316         {
317             //假定第一个构造函数的调用在线程启动起
318             if (!__node_allocator_lock_initialized)
319             {
320                 InitializedCriticalSection(&__node_allocator_lock);
321                 __node_allocator_lock_initialized = true;
322             }
323         }
324     private:
325 # endif
326
327         class lock
328         {
329             public:
330                 lock() { __NODE_ALLOCATOR_LOCK; }
331                 ~lock() { __NODE_ALLOCATOR_UNLOCK; }
332         };
333         friend class lock;
334     public:
335         //n必须大于0
336         static void * allocate(size_t n)
337         {
338             obj * __VOLATILE * my_free_list;
339             obj * __RESTRICT result;
340
341             //需要分配的大小大于二级配置器的__MAX_BYTES,直接使用第一级配置器
342             if (n > (size_t) __MAX_BYTES)
343             {
344                 return(malloc_alloc::allocate(n));
345             }
346             my_free_list = free_list + FREELIST_INDEX(n); //找到比需要分配的大小大,且最接近的大小块所在的链表所在free_list数组中的位置
347
348             //如果支持线程,定义lock
349 #        ifndef _NOTHREADS
350             lock lock_instance;
351 #        endif
352             result = *my_free_list;  //取出找的对应链表的指向第一个节点的指针
353             if (result == 0)    //对应的链表中没有剩余未分配的节点区块
354             {
355                 void *r = refill(ROUND_UP(n));    //再从内存池中分配一批,需求大小的区块(实际大小是请求大小上调至8的倍数后的数值),
356                                 //然后,放入对应链表,待分配给请求
357                 return r;
358             }
359             //如果对应大小区块的链表中不为空,还有待分配的区块,取出第一个节点
360             *my_free_list = result -> free_list_link;
361             return (result);
362         };
363
364         //p不可以是0
365         static void deallocate(void *p, size_t n)
366         {
367             obj *q = (obj *)p;
368             obj * __VOLATILE * my_free_list;
369
370             //大于区块大小上限的,直接调用第一级配置器释放
371             if (n > (size_t) __MAX_BYTES)
372             {
373                 malloc_alloc::deallocate(p, n);
374                 return;
375             }
376             my_free_list = free_list + FREELIST_INDEX(n);
377             //需要修改my_free_list,如果支持线程,那么需要加上互斥锁
378 #            ifndef _NOTHREADS
379             lock lock_instance;
380 #            endif
381
382             //头插法,插入对应大小的区块链表
383             q -> free_list_link = *my_free_list;
384             *my_free_list = q;
385             //lock是静态对象,到此,将自动析构销毁,在其析构函数中,会释放锁
386         }
387
388         static void *reallocate(void *p, size_t old_sz, size_t new_sz);
389
390 };
391
392 typedef __default_alloc_template<__NODE_ALLOCATOR_THREADS, 0> alloc;
393 typedef __default_alloc_template<false, 0> single_client_alloc;
394
395
396 // 我们从大的chunks中分配内存,是为了避免使用malloc太频繁了
397 // 假设size已经适当上调至8的倍数
398 // 我持有allocation lock
399 // 注意参数objs 是pass by reference
400 template <bool threads, int inst>
401 char *
402 __default_alloc_template<threads, inst>::chunk_alloc(size_t size, int& nobjs)
403 {
404     char * result;
405     size_t total_bytes = size * nobjs;
406     size_t bytes_left = end_free - start_free;    //内存池剩余空间
407
408     if (bytes_left >= total_bytes)
409     {
410         //内存池中剩余的空间足够满足需求量
411         result = start_free;
412         start_free += total_bytes;
413         return(result);
414     }
415     else if (bytes_left >= size)
416     {
417         //内存池剩余空间不能完全满足需求量,但足够供应一个及以上的区块
418         nobjs = bytes_left/size;
419         total_bytes = size * nobjs;
420         result = start_free;
421         start_free += total_bytes;
422         return (result);
423     }
424     else
425     {
426         //内存池连一个区块的大小都无法满足
427         size_t bytes_to_get = 2 * total_bytes + ROUND_UP(heap_size >> 4);
428         //以下试着让内存池中的残余零头还有利用价值
429         if (bytes_left > 0)
430         {
431             //内存池中内还有一些零头,先配给适当的free list
432             //首先寻找适当的free list
433             obj * __VOLATILE * my_free_list =
434                     free_list + FREELIST_INDEX(bytes_left);
435
436             //调整free list,将内存池中残余的空间编入
437             ((obj *)start_free) -> free_list_link = *my_free_list;
438             *my_free_list = (obj *)start_free;
439         }
440
441         //配置heap空间,用来补充内存池
442         start_free = (char *)malloc(bytes_to_get);
443         if (0 == start_free)
444         {
445             //如果heap空间不足,malloc()失败
446             int i;
447             obj * __VOLATILE *my_free_list, *p;
448             //试着检视我们手上的东西。这不会造成伤害。我们不打算尝试配置
449             //较小的区块,因为那在多线程机器上容易导致灾难
450             //以下搜索适当的free list
451             //所谓适当是指“尚有未用区块,且区块够大”之free list
452             for (i = size; i <= __MAX_BYTES; i += __ALIGN)
453             {
454                 my_free_list = free_list + FREELIST_INDEX(i);
455                 p = *my_free_list;
456                 if (0 != p)
457                 {
458                     //free list内尚有未用区块
459                     //调整free list以释放出未用的区块到内存池
460                     *my_free_list = p -> free_list_link;
461                     start_free = (char *)p;
462                     end_free = start_free + i;
463                     // 此时内存池已经有内存了
464                     // 递归调用自己,为了修正objs
465                     return chunk_alloc(size, nobjs);
466                     //注意,任何残余的零头终将被编入适当的free list中备用
467
468                 }
469             }
470             end_free = 0;  //如果出现意外(山穷水尽,到处都没有内存可用了)
471             //调用第一级配置器,看看out-of-memory机制能否尽点力
472             start_free = (char *)malloc_alloc::allocate(bytes_to_get);
473             //这会导致抛出异常,或内存不足的情况获得改善
474         }
475         heap_size += bytes_to_get;
476         end_free = start_free + bytes_to_get;
477         //递归调用自己,为了修正objs
478         return chunk_alloc(size, nobjs);
479     }
480 }
481
482
483
484 // 返回一个大小为n的对象,并且有时候会为适当的free list 增加节点
485 // 假设n已经适当上调至8的倍数
486 // 我们持有allocation lock
487 template <bool threads, int inst>
488 void* __default_alloc_template<threads, inst>::refill(size_t n)
489 {
490     int nobjs = 20;  //默认一次分配20个需求大小的区块
491     char * chunk = chunk_alloc(n, nobjs); //chunk是分配的空间的开始地址,令其类型为char *,主要是因为一个char的大小正好是一个byte
492     obj * __VOLATILE *my_free_list;
493     obj * result;
494     obj * current_obj, * next_obj;
495     int i;
496
497     //如果只获得一个区块,这个区块就分配给调用者,free list 无新节点
498     if (1 == nobjs) return chunk;
499     //否则准备调整free list,纳入新节点
500     my_free_list = free_list + FREELIST_INDEX(n);
501
502     //以下在chunk空间内建立free list
503     result = (obj *)chunk;  //这一块准备返回给客端
504     // 以下导引free list 指向新配置的空间(取自内存池)
505
506     //由于chunk是char*,所以加上n,就表示走过n个char,
507     //一个char正好是一个byte,所以chunk+n现在指向第二个区块
508     *my_free_list = next_obj = (obj *)(chunk + n);
509     for (i = 1; ; ++i)
510     {
511         // 从1开始,因为第0个将返回给客端
512         current_obj = next_obj;
513         // 每次移动n个char,正好是n个byte,所以正好指向下个区块
514         next_obj = (obj *)((char *)next_obj + n);
515         if (nobjs - 1 == i)
516         {
517             // 已经遍历完,此时next_obj指向的内存已经超出我们分配的大小了
518             // 不属于我们的内存
519             current_obj -> free_list_link = 0;
520             break;
521         }
522         else
523         {
524             current_obj -> free_list_link = next_obj;
525         }
526     }
527     return result;
528 }
529
530
531 template<bool threads, int inst>
532 void*
533 __default_alloc_template<threads, inst>::reallocate(void *p,
534                                                     size_t old_sz,
535                                                     size_t new_sz)
536 {
537     void * result;
538     size_t copy_sz;
539
540     if (old_sz > (size_t) __MAX_BYTES && new_sz > (size_t) __MAX_BYTES)
541     {
542         return realloc(p, new_sz);
543     }
544     if (ROUND_UP(old_sz) == ROUND_UP(new_sz)) return p;
545     result = allocate(new_sz);
546     copy_sz = new_sz > old_sz ? old_sz : new_sz;
547     memcpy(result, p, copy_sz);
548     deallocate(p, old_sz);
549     return result;
550
551 }
552
553 #ifdef __STL_PTHREADS
554     template <bool threads,int inst>
555     pthread_mutex_t
556     __default_alloc_template<threads, inst>::__node_allocator_lock
557         = PTHREAD_MUTEX_INITIALIZER;
558 #endif
559
560 #ifdef __STL_WIN32THREADS
561         template <bool threads, int inst> CRITICAL_SECTION
562         __default_alloc_template<threads, inst>::__node_allocator_lock;
563
564         template <bool threads, int inst> bool
565         __default_alloc_template<threads, inst>::__node_allocator_lock_initialized
566             = false;
567 #endif
568
569 //省略了通用lock的实现(即不使用pthread,也没有win32thread)
570
571 template <bool threads, int inst>
572 char *__default_alloc_template<threads, inst>::start_free = 0; //设置初始值
573
574 template <bool threads, int inst>
575 char *__default_alloc_template<threads, inst>::end_free = 0; //设置初始值
576
577 template <bool threads, int inst>
578 size_t __default_alloc_template<threads, inst>::heap_size = 0; //设置初始值
579
580 //初始化16种大小的区块链表为空
581 template <bool threads, int inst>
582 typename __default_alloc_template<threads, inst>::obj * __VOLATILE
583 __default_alloc_template<threads, inst>::free_list[
584 # ifdef __SUNPRO_CC
585     __NFREELISTS
586 # else
587     __default_alloc_template<threads, inst>::__NFREELISTS
588 # endif
589 ] = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, };
590
591 # ifdef __STL_WIN32THREADS
592   // Create one to get critical section initialized.
593   // We do this onece per file, but only the first constructor
594   // does anything.
595   static alloc __node_allocator_dummy_instance;
596 # endif
597
598 # endif /* ! __USE_MALLOC */
599
600 #if defined(__sgi) && !defined(__GNUC__) && (_MIPS_SIM != _MIPS_SIM_ABI32)
601 #pragma reset woff 1174
602 #endif
603
604 __STL_END_NAMESPACE
605
606 #undef __PRIVATE
607
608 #endif /* __SGI_STL_INTERNAL_ALLOC_WJZH_H */
609
610 //End

  

时间: 2024-12-21 16:44:42

自己动手实现STL 01:内存配置器的实现(stl_alloc.h)的相关文章

STL容器内存配置器

本系列文章更多是笔记形式,希望能在总结过程中将一些东西理顺.难免出错,欢迎指正. STL六大功能组件: 1.容器(containers):2.算法(algorithm):3.迭代器(iterator):4.仿函数(functors):5.配接器(adapters):6.配置器(allcators). 各个功能组件间存在交互关系,这里不涉及这些内容,本篇文章讨论容器的内存配置. 首先,容器用来存放数据,那么存放数据之前必须向系统申请内存资源.我们知道c++中通常用(::operator new/:

内存配置器

stl中内存配置器分为两级:第一级配置对象超过128B的内存,第二级配置对象小于128B的内存,stl默认采用第二级内存配置器,因为如果对象大于128B,则第二级内存配置器会自动调用第一级内存配置器. 第一级内存配置器简单的对malloc,realloc,free的封装 第二级内存配置器采用内存池管理内存,并用16个链表管理回收到的空闲内存块,当分配内存时,首先找到合适位置的空闲链表,若链表上有空闲内存块,则直接摘下空闲内存块供使用;否则就需要向内存池申请新的空闲内存块,若内存池都没有剩余内存了

SGI STL内存配置器存在内存泄漏吗?

阅读了SGI的源码后对STL很是膜拜,很高质量的源码,从中学到了很多.温故而知新!下文中所有STL如无特殊说明均指SGI版本实现. STL 内存配置器 STL对内存管理最核心部分我觉得是其将C++对象创建过程分解为构造.析构和内存分配.释放两类操作分离开来!摆脱了对频繁调用new或malloc函数想操作系统申请空间而造成的低效.其中析构操作时对具有non-trival.trival 析构函数的class区别对待也提高了效率.SGI 的两级配置器结构属于锦上添花. STL内存配置器有没有内存泄漏?

SGI STL内存配置器(一):内存泄漏?

阅读了Alexander大神的SGI STL源码,膜拜,很高质量的源码,获益匪浅.温故而知新!下文中所有STL如无特殊说明均指SGI版本实现. STL 内存配置器 STL对内存管理最核心部分我觉得是它将C++对象创建过程分解为构造.析构和内存分配.释放!摆脱了由于频繁调用new或malloc函数向操作系统申请空间而造成的低效.其中析构操作时对具有non-trival.trival 析构函数的class区别对待也提高了效率.至于SGI的两级配置器结构则属于锦上添花的类型. STL两级结构的内存配置

STL内存配置器

一.STL内存配置器的总体设计结构 1.两级内存配置器:SGI-STL中设计了两级的内存配置器,主要用于不同大小的内存分配需求,当需要分配的内存大小大于128bytes时, 使用第一级配置器,否则使用第二级配置器:对于小块的内存的分配使用第二级配置器使用分配与释放内存块的效率更高,时间复杂度为O(1): 2.两级配置器的优点: (1)使用两级配置器主要是为了避免太多的小块的内存申请导致内存空间中的内存碎片问题: (2)使用第二级配置器也可以避免太多小块内存分配导致的内存空间浪费问题,因为在申请内

【STL】空间配置器剖析(二)

上篇文章点击打开链接主要对于对象的构造含和析构进行了主要说明,这篇文章将对对象构造前的内存配置和对象析构后的空间释放进行深入探索. 好的,话不多说马上进入是正文: 对对象构造前的内存配置和对象析构后的空间释放,由<stl_alloc.h>负责,SGI对此的设计哲学如下: 向system heap要求空间. 考虑多线程的状态 考虑内存不足的应变措施 考虑过多的"小型区块"可能造成的内存碎片问题 C++的内存配置的基本操作是:operator new(),内存释放的基本操作是o

c++ stl 内存配置器

在STL中,Memory Allocator 处于最底层的位置,为一切的 Container 提供存储服务,是一切其他组件的基石.对于一般使用 STL 的用户而言,Allocator 是不可见的,如果需要对 STL 进行扩展,如编写自定义的容器,就需要调用 Allocator 的内存分配函数进行空间配置. 在C++中,一个对象的内存配置和释放一般都包含两个步骤,对于内存的配置,首先是调用operator new来配置内存,然后调用对象的类的构造函数进行初始化:而对于内存释放,首先是调用析构函数,

STL 之 空间配置器(allocator)

一.SGI 标准的空间配置器,std::allocator SGI也定义了一个符合部分标准,名为allocator的配置器,但是它自己不使用,也不建议我们使用,主要原因是效率不佳. 它只是把C++的操作符::operator new和::operator delete做了一层简单的封装而已. 二.SGI 特殊的空间配置器,std::alloc 由于SGI 标准的空间配置器只是把C++的操作符::operator new和::operator delete做了一层简单的封装,没有考虑到任何效率上的

(一)STL剖析——空间配置器

Alloc.h //Alloc.h负责内存空间的配置与释放 //Construct.h负责对象内容的构造与析构 //这两个头文件在memory文件中包含 #pragma once typedef void(*HANDLER_FUNC)(); //一级空间配置器 template <int inst> class __MallocAllocTemplate  { public: static void* Allocate(size_t size)//静态成员函数 { void* result =