2.1 空间配置器(allocator)

空间配置器隐藏在一切组件之后。

1.1 设计一个简单的空间配置器

  • 根据STL的规范,以下是allocator的必要接口:  
    allocator::value_type
    allocator::pointer
    allocator::const_pointer
    allocator::reference
    allocator::const_reference
    allocator::size_type
    allocator::difference_type
    allocator::rebind // 一个嵌套的(nested)class template。class rebind<U>拥有唯一成员other,是一个typedef,代表alloctor<U>
    allocator::allocator() // default constructor
    allocator::allocator(const allocator&) // copy constructor
    template <class U>allocator::allocator(const allocator<U>&) //泛化的default constructor
    allocator::~allocator() // default constructor
    pointer allocator::address(reference x) const // 返回某个对象的地址。a.address(x)等同于&x
    const_pointer allocator::address(const_reference x) const // 返回某个const对象的地址。a.address(x)等同于&x
    pointer allocator::allocate(size_type n, const void* = 0) // 配置空间,足以存储n个T对象。第二个参数是个提示,可能会用来增进locality,可忽略
    void allocator::deallocate(pointer p, size_type n) // 归还先前配置的空间
    size_type allocator::max_size() const // 返回可成功配置的最大量
    void allocator::construct(pointer p, const T& x) // 等同于new(const void*) p) T(x)
    void allocator::destroy(pointer p) // 等同于p->~T()

1.2 一个简单的allocator源代码

// filename : qyalloc.h
#ifndef __QYALLOC__
#define __QYALLOC__

// placement new是operator new的一个重载版本,只是我们很少用到它。如果你想在已经分配的内存中创建一个对象,使用new是不行的。也就是说placement new允许你在一个已经分配好的内存中(栈或堆中)构造一个新的对象。原型中void*p实际上就是指向一个已经分配好的内存缓冲区的的首地址。
// placement new的作用就是:创建对象(调用该类的构造函数)但是不分配内存,而是在已有的内存块上面创建对象。用于需要反复创建并删除的对象上,可以降低分配释放内存的性能消耗。请查阅placement new相关资料。
#include <new>          // placement new 要包含此文件,声明了一个void *operator new( size_t, void *p ) throw()  { return p; }
#include <cstddef>      // for ptrdiff_t, size_t
#include <cstdlib>      // for exit()
#include <climits>      // for UINT_MAX
#include <iostream>     // for cerr

namespace QY{
    // 分配空间(operator new)
    template <class T>
    inline T* _allocate(ptrdiff_t size, T*){
        std::set_new_handler(0);
        T *tmp = (T*)(::operator new((size_t)(size * sizeof(T))));
        if (tmp == 0)
        {
            std::cerr << "out of memory" << std::endl;
            exit(1);
        }
        return tmp;
    }
    
    // 回收空间(operator delete)
    template <class T>
    inline void _deallocate(T* buffer){
        ::operator delete(buffer);
    }
    
    // 在指定内存上构造一个对象(new(pMyClass)MyClass();)
    template <class T1, class T2>
    inline void _construct(T1* p, const T2& value){
        new(p) T1(value);       // 创建。placement new. 调用 ctor of T1, 即new(pMyClass)MyClass();
    }
    
    // 析构对象
    template<class T>
    inline void _destroy(T* ptr){
        ptr->~T();      
    }
    
    // 按allocator标准,定义结构
    template <class T>
    class allocator{
        public:
            typedef T           value_type;
            typedef T*          pointer;
            typedef const T*    const_pointer;
            typedef T&          reference;
            typedef const T&    const_reference;
            typedef size_t      size_type;
            typedef ptrdiff_t   difference_type;
            
            // 重新绑定分配器(rebind allocator of type U)
            template <class U>
            struct rebind
            {
                typedef allocator<U> other;
            };
            
            pointer allocate(size_type n, const void* hint=0){
                return _allocate((difference_type)n, (pointer)0);
            }
            
            void deallocate(pointer p, size_type n){
                _deallocate(p);
            }
            
            void construct(pointer p, const T& value){
                _construct(p, value);
            }
            
            void destroy(pointer p){
                _destroy(p);
            }
            
            pointer address(reference x){
                return (pointer)&x;
            }
            
            const_pointer address(const_reference x){
                return (const_pointer)&x;
            }
            
            size_type max_size() const{
                return size_type(UINT_MAX / sizeof(T));
            }            
    };
}   // end of namespace QY

#endif  // __QYALLOC__

1.3 使用这个allocator

#include "qyalloc.h"
#include <vector>
#include <iostream>
using namespace std;

int main(){
    int ia[5] = {0, 1, 2, 3, 4};
    unsigned int i;
    
    vector<int, QY::allocator<int> > iv(ia, ia+5);
    for(i=0; i<iv.size(); i++)
        cout << iv[i] << ' ';
    cout << endl;   
    
    return 0; 
}

1.4 SGI标准的空间配置器(std::allocator)

符合部分标准,效率不佳,不建议使用。

1.5 SGI特殊的空间配置器(std::alloc)

 class Foo{ ... };
 Foo* pf = new Foo;    // 配置内存,然后构造对象
 delete pf;    // 将对象析构,然后释放内存

new: (1)调用 ::operator new 配置内存;
     (2)调用 Foo::Foo() 构造对象内容。
  delete: (1)调用 Foo::~Foo() 将对象析构;
     (2)调用 ::operator delete 释放内存。
  为了精密分工,STL allocator 将两阶段操作区分开来。
  alloc::allocate()负责内存配置操作;
  alloc::deallocate()负责内存释放操作;
  ::construct()负责对象构造操作;
  ::destroy()负责对象析构操作。


1.6 构造和析构基本工具:construct() 和 destroy()

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352

推荐阅读更多精彩内容