简介

forwrad_list字面意思为前向列表,但实际上它是一种单向列表,只能从单一方向遍历。

单向链表实现

forward_list内部是用单向列表实现的,并且设计该库的时候就是以近乎手写的单向链表的运行效率(时间上和空间上)为目的的。 这导致了它是唯一一个C++标准库容器中没有size成员函数的容器, 因为维护这样一个信息会造成效率上的轻微损失。

作为单向链表,它有以下几个属性:

  • 潜在可能的非连续内存分配
  • 线性时间的元素位置获取
  • 常数时间的元素插入、删除、移动

与list的共性

因为它们的内部实现都是通过链表(不管是单链表还是双链表),所以forwrd_list类模版具有一些与list相同的特殊函数:

  • splice_after
  • remove、remove_if
  • unique
  • merge
  • sort
  • reverse

与list的差异

明显的,forward_list是单向链表,内部只维护了单向遍历的信息。 因此,forward_list的迭代器是前向迭代器(forward intertor)。

除此之外,它们的插入操作也有明显的不同,具体体现在传入的迭代器上:

  • list::insert:在传入的迭代器之前插入。
  • forward_list::insert:在传入的迭代器之后插入。

我们来看,为什么标准库要放弃接口语义的一致性,采用不一样的接口设计。

节点修改策略

单向 VS 双向

典型的链表分单向和双向,基本上每一个方向上都需要额外的空间来保存顺序信息(即前一个和后一个元素)。 双向链表保存有两个方向的链接, 而单向链表只有一个方向。

节点修改

当涉及到链表节点的修改时,例如插入、移动等,问题就变得明显了:对于链表来说,修改单个节点实际上涉及到3个节点的信息----当前节点、前驱节点lis、后继节点(修改前后节点的前后节点的链接信息是必要的)。

但是,可以更简单点:双向链表的前驱和后继信息可以间接地通过当前节点得到,因为当前节点有足够的信息 。也就是说,在修改节点时,我只要传入该节点就可以了;而单向链表单个节点可以通过链接信息获得后继节点

传入前驱节点迭代器

实际上,forward_list库做得更好,它的相关接口只需要我们传入一个节点迭代器。 这些接口包括(分别对应常见的非after版本):

  • insert_after
  • emplace_after
  • erase_after

那么这怎么做到的呢? forward_list的策略是----传入前驱节点的迭代器。 这样一来,所有需要的节点信息都能够直接间接地得到。 Great!

但是这也有一个问题:怎么在头部插入一个元素呢?毕竟头节点没有前驱节点。 哈哈,这就是为什么这个类模版提供了两个非常奇怪的函数, 它们是专门为上述三个函数提供服务的(-_-):

  • before_begin
  • before_cbegin

所以现在你也知道了,为什么随笔开头的示意图有一个灰色的节点了吧。ヽ(^o^)ノ

C++ 头文件系列(forward_list)的更多相关文章

  1. C++ 头文件系列(array)

    注意,该头文件仅在C++11中标准才开始出现. 简介 与语言内置的数组一样, array类模版支持几乎所有内置数组包含的特性: 顺序的(sequence) 内存连续的(contiguous stora ...

  2. C++ 头文件系列(queue)

    简介 这个头文件定义了两个跟队列有关的类----quque.priority_queue,分别实现的是队列 和 优先队列这两个概念. 但是与这两个类模版与其它类模版(vector.array等)最大的 ...

  3. C++ 头文件系列(stack)

    简介 该头文件只含有一个类模版stack, 它实现栈的概念,是一个容器适配器(说实话,在写这篇随笔之前我都不知道有这么个类模版). 栈 栈只有一个重要的特性: LIFO(last-in first-o ...

  4. C++ 头文件系列(vector)

    简介 vector头文件包含vector的类模版以及该模版的显示特化版本vector< bool >. vector是C++容器库中非常通用的一种容器,如果你不知道该决定使用哪一种容器,或 ...

  5. C++ 头文件系列(map)

    简介 该头文件包含两个概念相似的容器----map.multimap. 而这两个容器反映的概念就是 映射. 这两个容器 相同 的属性有: 关联性 映射 动态增长 键(Key)唯一性 这两个不相同的属性 ...

  6. C++ 头文件系列(unordered_map、unordered_set)

    简介 很明显,这两个头文件分别是map.set头文件对应的unordered版本. 所以它们有一个重要的性质就是: 乱序 如何乱序 这个unorder暗示着,这两个头文件中类的底层实现----Hash ...

  7. C++ 头文件系列(iterator)

    简介 该头文件围绕迭代器展开,定义了一系列与迭代器有关的概念,但最最最重要的一点就是----它和其它容器一起实现了C++容器的Iterator设计模式. Iterators are a general ...

  8. C++ 头文件系列 (algorithm)

    简介 algorithm头文件是C++的标准算法库,它主要应用在容器上. 因为所有的算法都是通过迭代器进行操作的,所以算法的运算实际上是和具体的数据结构相分离的 ,也就是说,具有低耦合性. 因此,任何 ...

  9. C++ 头文件系列 (bitset)

    简介 该头文件有关位集,实际上是vector 位 位本质上对应bool的概念,只有0或1,true或false两种对立的值. 但很可惜,字节才是机器上最小的存储单元,所以bool基本上是由一个字节大小 ...

随机推荐

  1. ural1090 In the Army Now

    In the Army Now Time limit: 1.0 secondMemory limit: 64 MB The sergeant ordered that all the recruits ...

  2. java采用Apache FileUpload组件实现上传

    可见:FileItemSteam(servletFileUpload.getItemIterator(httpServletRequest))速度要快于FileItem(servletFileUplo ...

  3. 黑科技--位集--bitset

    自从上次网赛发现这么个东西之后,深深地感受到了bitset的强大,0.0. 正常的bool占用1字节空间,bitset可以把这个缩到1bit,空间上8倍优化.正常用起来可能会跟位运算状态压缩类似,但是 ...

  4. Quick Cocos2dx controller的初步实现

    很久没有记笔记了,今天记一下,最近都在瞎忙活,都不知道自己干了些啥. 我的Controller是在官方的mvc sample的里面的PlayerDualController上更改的,所以很多地方还没来 ...

  5. clip实现圆环进度条

    效果主要通过clip和transform:rotate实现 把圆环分为左右两个部分,通过角度旋转对图片剪切旋转角度<=180度的时候之旋转右边,当大于180度时右边固定旋转180度的同时旋转左边 ...

  6. (简单) POJ 3076 Sudoku , DLX+精确覆盖。

    Description A Sudoku grid is a 16x16 grid of cells grouped in sixteen 4x4 squares, where some cells ...

  7. mysql优化———第二篇:数据库优化调整参数

    摘要 参数调优内容: 1. 内存利用方面 2. 日志控制方面 3.文件IO分配,空间占用方面 4. 其它相关参数 一  摘要 通过参数提高MYSQL的性能.核心思想如下:         1 提高my ...

  8. js之动态加载等待图像地址汇总

    Ajax火啊,火到了居然Loading Icons都有很多人专门提供的地步.下面是我同事给我介绍的一些提供Ajax Activity Indicators的网站,共享给大家,以便让我们的Ajax应用具 ...

  9. iOS开发——工厂模式

    工厂模式很好用,为表诚意,我直接搞个实用的例子放这,解析一个订单的数据,并且这个订单里面可能不止一件商品的做法. 还是直接上代码,不懂的地方,再提出来. 1.在MyOrderDeals.h文件中 #i ...

  10. 我推荐的一些iOS开发书单

    文/叶孤城___(简书作者)原文链接:http://www.jianshu.com/p/2fa080673842著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”. 上次发了一下比较不错的i ...