内存分配和释放

STL中有两个分配器,一级分配器和二级分配器,默认使用二级分配器,使用二级分配器分配大内存时会调用一级分配器去执行,一级分配器使用malloc和free分配和释放内存。如果分配小内存那么二级分配器会从内存池中进行查找,防止malloc/free的开销。

为了了解原理,不深挖细节,只实现一级分配器也是可以的:

class first_level_alloc {
public:
    static void* allocate(size_t n) {
        void* result = malloc(n); //直接使用malloc
        //todo: out of memory handler
        return result;
    }
    static void deallocate(void* p, size_t) {
        free(p); //直接使用free
    }
};

一级分配器,直接调用malloc和free分配和释放内存。这里也没有处理分配失败的情况。

为了方便使用定义一个包装类:

template <typename T, typename Alloc>
class simple_alloc {
public:
    static T* allocate(size_t n) {
        return (0 == n) ? nullptr : static_cast<T *>(Alloc::allocate(n * sizeof(T)));
    }
    static T* allocate() {
        return static_cast<T *>(Alloc::allocate(sizeof(T)));
    }
    static void deallocate(T *p, size_t n) {
        if (0 != n) {
            Alloc::deallocate(p, n * sizeof(T));
        }
    }
    static void deallocate(T *p) {
        Alloc::deallocate(p, sizeof(T));
    }
};

对外使用这个包装类模板参数T指定要分配的对象类型,Alloc指定分配器,因为没有实现二级分配器,所以都是指定为一级分配器first_level_alloc。

对象的构造和析构

定义如下三个函数:

template <typename T>
inline void construct(T* p, const T& value) {
    new(p) T(value); //placement new
}
template <typename T>
inline void destroy(T *p) {
    p->~T();
}
//todo:low efficiency
template <typename ForwardIterator>
inline void destroy(ForwardIterator first, ForwardIterator last) {
    for (; first != last; ++first) {
        destroy(&*first);
    }
}
  1. void construct(T* p, const T& value):在p指向的位置用value拷贝构造T对象并返回。这里用到了placement new。
  2. void destroy(T *p):析构p指向处的T对象。
  3. void destroy(ForwardIterator first, ForwardIterator last):析构[first, last)区间的对象。这里没有考虑效率,直接使用for循环调用destroy。STL库中使用模板特例化,根据迭代器指向的类型有没有trivial destructor,执行不同的特例化版本。如果有trivial destructor,比如内置类型,那么什么也不用做。如果有non-trivial destructor才调用上述的那个版本。

traits要解决的问题

假如算法中要声明“迭代器所指类别”的变量,该怎么办?

内嵌类别声明解决非指针迭代器的情况

template <typename T>
struct MyIter { //模拟迭代器类型
typedef T value_type; //内嵌类别声明
T* ptr;
MyIter(T* p = 0) :ptr(p) {}
T& operator*() const {
return *ptr;
}
}; template <typename I>
typename I::value_type //返回类型为迭代器指向的类型
func(I ite) { //该函数传入一个指针,返回指针指向的值。
return *ite;
} int main() {
MyIter<int> ite(new int(8));
cout << func(ite);
}

MyIter模拟迭代器,T是迭代器所指的类型,通过在迭代器内typedef T value_type;后,就能用MyIter::value_type定义T类型的变量。

上面的方法解决了一部分问题,但是普通指针也是迭代器类型,我们没办法给指针应用上面的方法。比如上面的func,如果我们传入一个指针,肯定无法通过编译。

使用模板特例化解决普通指针的情况

template <typename T>
struct MyIter {
typedef T value_type;
T* ptr;
MyIter(T* p = 0) :ptr(p) {}
T& operator*() const {
return *ptr;
}
}; template <typename I>
struct iterator_traits { //针对普通迭代器的模板类
typedef typename I::value_type value_type;
}; template <typename I>
struct iterator_traits<I*> { //针对指针类型的模板特例化
typedef I value_type;
}; template <typename I>
typename iterator_traits<I>::value_type
func(I ite) { //该函数返回迭代器或这种指向的值
return *ite;
} int main() {
MyIter<int> it(new int(8));
int* ip = new int(8);
std::cout << func(ip) << std::endl;
std::cout << func(it);
}

这里定义了一个模板类iterator_traits,实际使用时iterator_traits<I>::value_type就是迭代器I所指的类型,如果是迭代器是指针类型,那么匹配的是itetraor_traits的特例化,iterator_traits<I>::value_type依然可以得到指针所指类型。

所以所谓的traits就是一个模板类和一系列模板特例化。通过这个模板类可以得到指针或者迭代器的相关类型。

同时如果一个迭代器类型如果想要和traits类配合使用需要在其内部通过typedef定义value_type类型。

迭代器相应类别

前面的迭代器所指类型value_type就是迭代器的相关类别之一,除了迭代器所指类型,还有几个迭代器相关类型。

  1. value_type:迭代器所指类型,上一节已经讲过了。
  2. difference type:用来表示两个迭代器之间的距离。
  3. reference type:迭代器所指类型的引用类型。
  4. pointer type:迭代器所指类型的指针类型。
  5. iterator_category:迭代器的类别。

STL-空间配置器、迭代器、traits编程技巧的更多相关文章

  1. STL空间配置器那点事

    STL简介 STL(Standard Template Library,标准模板库),从根本上说,STL是一些“容器”的集合,这些“容器”有list,vector,set,map等,STL也是算法和其 ...

  2. 【陪你系列】5 千字长文+ 30 张图解 | 陪你手撕 STL 空间配置器源码

    大家好,我是小贺. 点赞再看,养成习惯 文章每周持续更新,可以微信搜索「herongwei」第一时间阅读和催更,本文 GitHub https://github.com/rongweihe/MoreT ...

  3. 【转】STL空间配置器

    STL空间配置器(allocator)在所有容器内部默默工作,负责空间的配置和回收.STL标准为空间配置器定义了标准接口(可见<STL源码剖析>P43).而具体实现细节则由各编译器实现版本 ...

  4. stl空间配置器线程安全问题补充

    摘要 在上一篇博客<STL空间配置器那点事>简单介绍了空间配置器的基本实现 两级空间配置器处理,一级相关细节问题,同时简单描述了STL各组件之间的关系以及设计到的设计模式等. 在最后,又关 ...

  5. STL空间配置器

    1.什么是空间配置器? 空间配置器负责空间配置与管理.配置器是一个实现了动态空间配置.空间管理.空间释放的class template.以内存池方式实现小块内存管理分配.关于内存池概念可以点击:内存池 ...

  6. 咬碎STL空间配置器

    STL空间配置器 一.开场白: 给我的感觉就是,了解是空间配置器的功能,是那么的明了:在看原理,我还是很开心:接下来是360度大转变: 那么长的变量或者函数命名.那么多的宏.不爽,不过,遇上我这种二货 ...

  7. STL空间配置器解析和实现

    STL空间配置器的强大和借鉴作用不言而喻,查阅资料,发现了Dawn_sf已经对其有了极其深入和详细的描述,所以决定偷下懒借用其内容,只提供自己实现STL空间配置器的源码,具体解析内容参考:(一)STL ...

  8. STL——空间配置器(构造和析构基本工具)

    以STL的运用角度而言,空间配置器是最不需要介绍的东西,它总是隐藏在一切组件(更具体地说是指容器,container)的背后,默默工作,默默付出.但若以STL的实现角度而言,第一个需要介绍的就是空间配 ...

  9. STL空间配置器、vector、list、deque、map复习

    本文写于2017-03-03,从老账号迁移到本账号,原文地址:https://www.cnblogs.com/huangweiyang/p/6440830.html STL的六大组件:容器.算法.迭代 ...

  10. STL——空间配置器(SGI-STL)

    一. 空间配置器标准接口 参见<STL源码剖析>第二章-2.1.<memory>文件. 二.具备次配置力的SGI空间配置器 1. SGI STL的配置器与众不同,也与标准规范不 ...

随机推荐

  1. 全分布式的Hadoop初体验

    背景 之前的时间里对 Hadoop 的使用都是基于学长所搭建起的实验环境的,没有完整的自己部署和维护过,最近抽时间初体验了在集群环境下装机.配置.运行的全过程,梳理总结到本文中. 配置 内存:8G C ...

  2. JScript分割字符串

    作者:朱金灿 来源:http://blog.csdn.net/clever101 不废话了,直接用代码说明吧: try { var ss = new Array(); var str="12 ...

  3. spring4+springmvc+hibernate4 demo

    来自 CSDN . 其实下面的更好:加入了maven集成.Spring4 MVC Hibernate4集成 下面也是一篇讲的很详细的文章: hibernate4无法保存数据 而自己遇到的hiberna ...

  4. 局部QEventLoop帮助QWidget不消失(也就是有一个局部事件循环始终在运行,导致程序被卡住那里,但仍可以接受事件。说白了就是有一个while语句死活不肯退出,直到收到退出信号)

    熟悉的陌生人 Qt 是事件驱动的,所以当你用Qt的时候,几乎时时刻刻和 QEventLoop 打交道.,只是你可能没有意识到: QCoreApplicaton::exec() QApplication ...

  5. Theano 编程核心

    1. 求偏导.更新以及模型的训练 以 LogisticRegression 为例: 求损失函数关于参数的偏导: import theano.tensor as T g_W = T.gradient(c ...

  6. STL algorithm算法lexicographical_compare(30)

    lexicographical_compare原型: std::lexicographical_compare default (1) template <class InputIterator ...

  7. [转]完美解决)Tomcat启动提示At least one JAR was scanned for TLDs yet contained no TLDs

    一.文章前言    本文是亲测有效解决At least one JAR was scanned for TLDs yet contained no TLDs问题,绝对不是为了积分随便粘贴复制然后压根都 ...

  8. WPF编游戏系列 之二 图标效果

    原文:WPF编游戏系列 之二 图标效果        本篇将要实现图标的两个效果:1. 显示图标标签,2. 图标模糊效果.在上一篇中提到Image没有HTML <img>的Title属性( ...

  9. KVM虚拟化知识的一些笔记

    一.KVM介绍 KVM:运行在内核空间,提供CPU 和内存的虚级化,以及客户机的 I/O 拦截.Guest 的 I/O 被 KVM 拦截后,交给 QEMU 处理. QEMU:修改过的为 KVM 虚机使 ...

  10. .net与.net core学习目录

    .net C#调用python 模拟请求(模拟header/gzip解压/泛型) C#控制台关闭之前做一些操作 C# 元组.匿名对象.ref&out DataTable转换为Entity(反射 ...