迭代器分类

C++ STL 中根据移动能力将迭代器分成了 5 类:

  1. Input Iterator 输入迭代器,只支持 operator++ 操作。

  2. Output Iterator 输出迭代器,只支持 operator++ 操作。

  3. Forward Iterator 前向迭代器,只支持 operator++ 操作。

  4. Bidirectional Iterator 双向迭代器,支持 operator-- 和 operator++ 操作。

  5. Random Access Iterator 随机访问迭代器,不仅支持 operator-- 和 operator++ 操作,还可以与整数进行算术运算。

int main()
{
// 1.输入迭代器, 只能读缓冲区
// 这样会直接进入读取状态一直读取int, 直到状态出错或遇到EOF
// 比如输入了非数字, ctrl + d等等
istream_iterator<int> initer(cin);
int x = *initer; // 从缓冲区读取第一个数到变量中 // 2.输出迭代器, 只能写到某个地方
ostream_iterator<int> oiter(cout, " "); // 绑定到标准输出, 分隔符为空格字符串
vector<int> v{1, 2, 3, 4};
copy(v.begin(), v.end(), oiter); // 输出1 2 3 4
cout << endl; // 3.前向迭代器, unordered_xxx系列的关联式容器的迭代器就是这个类型的
// 大部分能读能写, set 系列由于键值合一, 不能够写
// map 中 key 不可更改, value 可以
unordered_map<int, char> m{{1, 'a'}, {2, 'b'}};
unordered_map<int, char>::iterator mit = m.begin();
while (mit != m.end()) {
mit->second = 'c'; // 可以修改 value
++mit; // 不支持 operator--
} // 4.双向迭代器, list 的迭代器就是该类型的
list<int> l{1, 2, 3};
list<int>::iterator lit = l.begin(); // 指向 1
++lit; // 指向 2
--lit; // 回到 1
*lit = 666; // 更改 // 5.随机访问迭代器, vector 和 deque 的迭代器就是该类型的
// 例如输出 vecor 的最后一个元素
vector<int>::iterator vit = v.begin();
cout << *(vit + v.size() - 1) << endl; // 支持与整数运算 return 0;
}

忽略了头文件,自行添加。

编译器多态

C++ STL 中很多算法会根据迭代器参数的类型来选择最优的实现方法,以 advance 为例,它的作用是将给定的迭代器移动指定步数,实现如下:

template<typename InputIterator, typename Distance>
void advance(InputIterator& it, Distance n)
{
// concept requirements -- taken care of in __advance
typename iterator_traits<InputIterator>::difference_type d = n;
std::__advance(i, d, std::__iterator_category(it));
}

可以看到它内部转调了 std::__advance,注意到它的第三个参数 std::__iterator_category(it)

该方法根据传入的迭代器返回一个迭代器类型对象,一共 5 种,迭代器类型命名类似于 xxx_iterator_tag。

然后std::__advance共有如下版本:

针对输入迭代器的:

template<typename InputIterator, typename Distance>
void __advance(InputIterator& it, Distance n, input_iterator_tag)
{
while (n--) ++it;
}

针对前向迭代器的:

template<typename InputIterator, typename Distance>
void __advance(InputIterator& it, Distance n, forward_iterator_tag)
{
__advance(it, n, input_iterator_tag()); // 直接转调
}

针对双向迭代器的:

template<typename InputIterator, typename Distance>
void __advance(InputIterator& it, Distance n, bidirectional_iterator_tag)
{
// 可以双向移动
if (n >= 0)
while (n--) ++it;
else
while (n++) --it;
}

针对随机访问迭代器的:

template<typename InputIterator, typename Distance>
void __advance(InputIterator& it, Distance n, random_access_iterator_tag)
{
it += n; // 直接加, 跳着走, 爽得一批
}

当我们调用 advance 时,编译器根据第三个参数(迭代器类型)选择最优模板。

C++迭代器种类与编译期间多态的更多相关文章

  1. C++编译期多态与运行期多态

    前言 今日的C++不再是个单纯的"带类的C"语言,它已经发展成为一个多种次语言所组成的语言集合,其中泛型编程与基于它的STL是C++发展中最为出彩的那部分.在面向对象C++编程中, ...

  2. Java编译时多态和运行时多态

    来源:https://blog.csdn.net/wendizhou/article/details/73733061 编译时多态:主要是方法的重载,通过参数列表的不同来区分不同的方法. 运行时多态: ...

  3. c++ 之 编译期多态&运行期多态

    编译时多态:程序运行前发生的事件 —— 函数重载.运算符重载 .模板  ——静态绑定 运行时多态:程序运行时发生的事件 —— 虚函数机制——动态绑定 template<typename T> ...

  4. 深入理解OOP(第一天):多态和继承(初期绑定和编译时多态)

    在本系列中,我们以CodeProject上比较火的OOP系列博客为主,进行OOP深入浅出展现. 无论作为软件设计的高手.或者菜鸟,对于架构设计而言,均需要多次重构.取舍,以有利于整个软件项目的健康构建 ...

  5. 读书笔记_Effective_C++_条款四十一:了解隐式接口和编译期多态

    从本条款开始,就进入了全书的第七部分:模板与泛型编程.模板与泛型在C++中是非常重要的部分,还记得本书第一章时,把C++视为一个联邦,它由四个州政府组成,其中一个政府就是模板与泛型了. 本条款是一个介 ...

  6. c#中运行时编译时 多态

    c#中运行时编译时 多态   public class aa { } public class bb:aa { } public class cc { public static void Main( ...

  7. 读书笔记 effective c++ Item 41 理解隐式接口和编译期多态

    1. 显示接口和运行时多态 面向对象编程的世界围绕着显式接口和运行时多态.举个例子,考虑下面的类(无意义的类), class Widget { public: Widget(); virtual ~W ...

  8. c++中如何定义编译期间常量,即这个常量可以用于定义数组下标

    在c++中,类里面的成员变量不仅仅可以被const修饰,还可以被static const修饰,此时一个内建类型(如int ,char ,long等)的static const 可以看做是一个编译期间的 ...

  9. Effective C++ Item 41 了解隐式接口和编译期多态

    本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie 经验:class 和 templates 都支持接口和多态. 对 classes 而言接口是 ...

  10. Effective C++ -----条款41:了解隐式接口和编译期多态

    classes和templates都支持接口(interface)和多态(polymorphism). 对classes而言接口是显式的(explicit),以函数签名为中心.多态则是通过virtua ...

随机推荐

  1. Kubernetes安装GitLab

    个人名片: 对人间的热爱与歌颂,可抵岁月冗长 Github‍:念舒_C.ying CSDN主页️:念舒_C.ying 个人博客 :念舒_C.ying Kubernetes安装GitLab Step 1 ...

  2. (C++) 初始化列表 std::initializer_list

    构造时直接使用初始化列表 T object { arg1, arg2, ... }; (1) T { arg1, arg2, ... } (2) new T { arg1, arg2, ... } ( ...

  3. Oracle 一对多将多行数据整理为一行

    在开发中一对多的关系经常用到,昨天遇到一个客户问题,要求将所有反馈任务的信息整理为一行展示并导出excel. 我数据库对应关系:通任务task对应feedback表多个反馈,也就是说每个任务都有很多条 ...

  4. 前后端分离开发工具YAPI部署记录

    之前公司说要建立起前后端分离开发模式,而我只是刚毕业,让我负责建立起这个规范 ,虽然刚毕业还没去大厂待过,对我来说是个挑战,只能按我理解和网上的方案进行建立.在 Google 和 github 搜了好 ...

  5. 第2-4-10章 规则引擎Drools实战(3)-保险产品准入规则

    目录 9.3 保险产品准入规则 9.3.1 决策表 9.3.2 规则介绍 9.3.3 实现步骤 9.3 保险产品准入规则 全套代码及资料全部完整提供,点此处下载 9.3.1 决策表 前面我们编写的规则 ...

  6. 【SQL进阶】【表默认值、自增、修改表列名、列顺序】Day02:表与索引操作

    一.表的创建.修改与删除 1.创建一张新表 [设置日期默认值.设置id自增] [注意有备注添加备注COMMENT] CREATE TABLE user_info_vip( id int(11) pri ...

  7. k8s篇-k8s集群架构及组件详解【史上最详细】

    O kubernetes简介 k8s是什么 k8s是一个可移植的.可扩展的开源平台,用于管理容器化的工作负载和服务,可以促进声明式配置和自动化. k8s能做什么 1)服务发现和负载均衡 Kuberne ...

  8. 使用selenium爬取淘宝

    一.出现的问题 前段时间在使用selenium对淘宝进行模拟登陆的时候,输入完正好和密码,然后验证码无论如何都不能划过去.找了好久,原来是因为selenium在浏览器中运        行的时候会暴露 ...

  9. python2和python3的区别(1)

    1.python2和python3的解释器的默认编码不同 python2解释器的编码默认用的是 ascii python3解释器的编码默认用的 utf-8 2.python2和python3输入的表示 ...

  10. 《HTTP权威指南》– 8.网关、Web机器人

    集成点:网关.隧道及中继 网关 网关(gateway): 资源和应用程序之间的粘合剂.应用程序可以(通过HTTP或其它已定义的接口)请求网关来处理某条请求,网关可以提供一条响应.网关可以向数据库发送查 ...