1.准备知识

  • typename用法
    • 用法1:等效于模板编程中的class
    • 用法2:用于显式地告诉编译器接下来的名称是类型名,对于这个区分,下面的参考链接中说得好,如果编译器不知道 T::bar 是类型名的话 T::bar * p可能就被理解成了T::bar 乘以p,T::bar & p可能就被理解成为了 T::bar 和p做逻辑与操作。 事实上,在模板编程时,如果传入的模板参数为T(T里面有模板参数的非独立名字bar),那么在不显示指定的话,c++或假定T::bar为变量名以消除歧义。
      • “A name used in a template declaration or definition and that is dependent on a template-parameter is assumed not to name a type unless the applicable name lookup finds a type name or the name is qualified by the keyword typename.”
    • 参考链接 : typename(wiki)
  • traits编程技法(内嵌型别使用)

traits意为榨汁机,是在SGI 版本的STL实现中被广泛使用到的一个编程技法,需要使用的原因是我们不能根据一个不知道类型的变量直接推导出它的型别,听起来好像很奇怪,也就是不存在这么一个函数: typeof(),举个例子说明如下:

  • 对于类型为 I 的变量 i ,I 可能是模板参数,如果对 I 来说 *操作有效,于是我们在程序中可以得到 *t ,然后需要将 *t 作为返回值返回,那么我们需要怎么做?下面问号的地方该怎么填?
    template<class I>

    ?? func(I i)

    {

        return *i;

    }

    我们经常需要在程序中获得迭代器相关的型别,由于迭代器与指针有某种类似的性质,所以迭代器所指型别很重要,经常需要在程序中使用,如上述问题之类的问题就被提出来了。这个问题的最终解决方案是---内嵌型别,对于刚才的函数,我们重新定义类 I ,在类 I 中声明一个内嵌型别,然后引用此内嵌型别,编译就可以通过,代码如下:

    //迭代器雏形

    template<class T>

    class myIter

    {

        //... T's other member ,ingored 

        //T * ptr

        //myIter(T * p=0):ptr(p){}

        typedef typename T value_type;

    }

     

    template<class I>

    typename I::value_type func(I i)

    {

        return *i;

    }

    到目前为止,问题好像解决了,根据上面的原理,我们可以定义traits,专门获取迭代器所指类型的型别:

    template<class I>

    struct iterator_traits

    {

        typedef typename I::value_type value_type;

        //...

    }

    但是在刚才分析时我们看到,要定义内嵌型别,首先 I 必须是一个类,对于有些类型的迭代器,可能就是一个原生指针,无法定义内嵌型别,这个时候就需要用到偏特化了,偏特化的内容在我的另一篇文章中讲过。在那边文章中我称作部分特化,这里遵从hj先生的翻译而已。偏特化只是提供另一份template定义,意思是对于某些特殊的类别作为模板参数时,使用不同的模板定义式,这些特殊的类别一般都是原生指针,const原生指针,或者是多个模板参数一样,比如对于原生指针,一个可以接受的traits特化版本就是:

    template<class I>

    struct iterator_traits<I*>

    {

       // typedef typename I value_type; (typename 可省略)

        typedef T value_type;

        //...

    }

    同样,对于const类型的原生指针,如果它作为迭代器,也必须定义一个偏特化的traits版本:

    template<class I>

    struct iterator_traits<const T *>

    {

        //attention : const tag is removed because a value weith const type is useless.

        typedef T value_type;

    }

2.迭代器相关型别

迭代器相关的型别(侯捷先生翻译为相应型别,个人感觉不太准确)有五种:

  • value type(所指型别)
  • 顾名思义,就是迭代器所指之物的型别,这个在上一节中作为例子讲过。

  • difference type(距离型别)

两个迭代器之间的距离

  • reference type(所指型别的引用型别)

对于可以改变所指对象内容的迭代器,当需要引用所指对象时,用到的型别就是引用型别。

  • pointer type(指针型别)

顾名思义

  • iterator catalog(迭代器类型型别)

这个实现起来稍微复杂点,之所以要有这个型别,主要是效率考虑,对不同的类型迭代器,施加不同的算法以保证最高效率,书中给出的例子是advanced的实现根据不同的迭代器类型的不同变种。根据移动特性和施加的操作,迭代器被分为五类:

  • input iterator
  • output iterator
  • foward iterator
  • bidirectional iterator
  • random acess iterator

这五种类型的iterator从上往下概念一次强化,从概念上来说,关系如下:

图中直线表示的是逐层强化的概念,而非继承关系,具体实现时,要表示不同型别的迭代器,在迭代器中要定义相应的非独立名称,这个非独立名称可以用来标识迭代器的种类,假设这五种非独立名称的定义如下:

struct input_iterator_tag{};

 

struct output_iterator_tag{};

 

struct forward_iterator_tag:input_iterator_tag{};

 

struct bidirectional_iterator_tag:forward_iterator_tag{};

 

struct random_acess_iterator_tag:bidirectional_iterator_tag

它的具体操作使用过程是这样的,对于具体实现算法的函数,有一个参数是tag类参数,调用时,调用一个封装函数,封装函数根据迭代器萃取出迭代器类型型别参数,然后作为参数传递到具体实现的算法中。如下:

template<class InputIterator,class Distance>

inline void __advance(InputIterator &i,Distance n,input_iterator_tag)

{

    while(n--)++i;

}

 

template<class ForwardIterator,class Distance>

inline void __advance(ForwardIterator &f,Distance n,forward_iterator_tag)

{

    //调用input_iterator_tag版本的

    __advance(f,n,input_iterator_tag());

}

 

template<class BidirectionalIterator,class Distance>

inline void __advance(Bidirectional &b,Distance n,bidirectional_iterator_tag)

{

    if(n>=0)while(n--)b++;

    else while(n++)b--;

}

 

template<class RandomAccessIterator,class Distance>

inline void __advance(RandomAccessIterator &r,Distance n,random_access_iterator_tag)

{

    r+=n;

}

 

然后加上一个对外封装的壳子如下,让编译器自动寻找最接近的实现函数:

template<class InputIterator,class Distance>

inline void advance(InputIterator &i,Distance n)

{

    __advance(i,n,iterator_traits<InputIterator>::iterator_catagory());

//    或许觉得__advance(i,n,InputIterator::iterator_catagory())说不定也可以通过,但是一个很显然的事情就是类似于原生指针这种形式就不可用了

}

 

3.std::iterator 的保证

任何iterator都应提供以上5种型别以供traits萃取,最好是继承下面的模板类:

template<class Catagory,class T,class Distance=ptrdiff_t,class Pointer=T*,class Reference=T&>

struct iterator{

    typedef Catagory iterator_catagory;

    typedef T value_type;

    typedef Distance difference_type;

    typedef Pointer pointer_type;

    typedef Reference reference_type;

}

4.总结

iterator是算法和数据结构之间的粘合剂,算法能统一地使用到stl的诸多数据结构中,iterator的设计功不可没,因此,对于stl中每一个容器,定好了底层的数据结构之后,设计相应的iterator就是接下来最重要的任务。(现在知道怎么回答iterator和指针的区别了吧)

STL源码分析读书笔记--第三章--迭代器(iterator)概念与traits编程技法的更多相关文章

  1. STL源码分析读书笔记--第5章--关联式容器

    1.关联式容器的概念 上一篇文章讲序列式容器,序列式容器的概念与关联式容器相对,不提供按序索引.它分为set和map两大类,这两大类各自有各自的衍生体multiset和multimap,的底层机制都是 ...

  2. STL源码分析读书笔记--第二章--空间配置器(allocator)

    声明:侯捷先生的STL源码剖析第二章个人感觉讲得蛮乱的,而且跟第三章有关,建议看完第三章再看第二章,网上有人上传了一篇读书笔记,觉得这个读书笔记的内容和编排还不错,我的这篇总结基本就延续了该读书笔记的 ...

  3. 重温《STL源码剖析》笔记 第三章

    源码之前,了无秘密. --侯杰 第三章:迭代器概念与traits编程技法 迭代器是一种smart pointer auto_Ptr 是一个用来包装原生指针(native pointer)的对象,声明狼 ...

  4. STL源码剖析读书笔记--第四章--序列式容器

    1.什么是序列式容器?什么是关联式容器? 书上给出的解释是,序列式容器中的元素是可序的(可理解为可以按序索引,不管这个索引是像数组一样的随机索引,还是像链表一样的顺序索引),但是元素值在索引顺序的方向 ...

  5. STL源码剖析读书笔记--第6章&第7章--算法与仿函数

    老实说,这两章内容还蛮多的,但是其实在应用中一点点了解比较好.所以我决定这两张在以后使用过程中零零散散地总结,这个时候就说些基本概念好了.实际上,这两个STL组件都及其重要,我不详述一方面是自己偷懒, ...

  6. STL源码剖析读书笔记之vector

    STL源码剖析读书笔记之vector 1.vector概述 vector是一种序列式容器,我的理解是vector就像数组.但是数组有一个很大的问题就是当我们分配 一个一定大小的数组的时候,起初也许我们 ...

  7. element-ui inputNumber、Card 、Breadcrumb组件源码分析整理笔记(三)

    inputNumber组件 <template> <!--@dragstart.prevent禁止input中数字的拖动--> <div @dragstart.preve ...

  8. 重温《STL源码剖析》笔记 第五章

    源码之前,了无秘密  ——侯杰 序列式容器 关联式容器 array(build in) RB-tree vector set heap   map priority-queue multiset li ...

  9. 重温《STL源码剖析》笔记 第四章

    源码之前,了无秘密  ——侯杰 序列式容器 关联式容器 array(build in) RB-tree vector set heap   map priority-queue multiset li ...

随机推荐

  1. PHP 投票练习

    重点:1.进度条的显示2.操作数据库<form action="chuli.php" method="post"> <?php include ...

  2. linux ps top 命令 VSZ,RSS,TTY,STAT, VIRT,RES,SHR,DATA的含义【转】

    转自:http://blog.csdn.net/zjc156m/article/details/38920321 http://javawind.net/p131 VIRT:virtual memor ...

  3. 堆的 两种实现 (数组和STL)

    基本思想: 两种操作都跟树的深度成正比,所以复杂度  O(log(n)) ; push():在向堆中插入数值时,首先在堆的末尾插入该数值,然后不断向上提直到没有大小颠倒为止. pop(): 从堆中取出 ...

  4. 在DirectX 中进行2D渲染

    http://flcstudio.blog.163.com/blog/static/756035392008115111123672/ 最近,我看到很多关于DirectX8在最新的API中摒弃Dire ...

  5. laravel重要概念和知识点

    Service Provider: 一个laravel service provider就是一个注册IoC container binding的类.实际上,laravel本身就自包含了一堆管理核心框架 ...

  6. Qt之模型/视图(自定义进度条)

    简述 在之前的章节中分享过关于QHeaderView表头排序.添加复选框等内容,相信大家模型/视图.自定义风格有了一定的了解,下面我们来分享一个更常用的内容-自定义进度条. 实现方式: 从QAbstr ...

  7. HDU 3448 Bag Problem

    这是一道搜索的背包题目 题意: 有n件物品从中最多选m件,使其总重量不超过v,求能获得的最大重量 有一个很重要的剪枝(是数据的问题还是这个剪枝本身很高效?): 如果重量最大m件物品都不超过v,则答案就 ...

  8. 51nod1757 大灾变

    能想到二分答案+最大流判断是否符合.但是不知道如何建图qaq.参考的是http://blog.csdn.net/fsss_7/article/details/52132046的建图方法 #includ ...

  9. 韦东山驱动视频笔记——3.字符设备驱动程序之poll机制

    linux内核版本:linux-2.6.30.4 目的:我们在中断方式的按键应用程序中,如果没有按键按下,read就会永远在那等待,所以如果在这个程序里还想做其他事就不可能了.因此我们这次改进它,让它 ...

  10. HDU 小明A+B 2096

    解题思路:So easy,练手速的,这种题很快1A,但是没有任何成就感.   转念一想,反正水题也就那几题,所以很快就会刷完. #include<cstdio> int main() { ...