STL之Iterator(迭代器)
概述
根据迭代器功能的不同,将迭代器分为以下几类:
| Iterator Category | Ability | Providers |
|---|---|---|
| Input iterator | Reads forward | istream |
| Output iterator | Writes forward | ostream, inserter |
| Forward iterator | Reads and writes forward | |
| Bidirectional iterator(双向迭代器) | Reads and writes forward and backward | list, set, multiset, map, multimap |
| Random access iterator | Reads and writes with random access | vector, deque string, array |
下面,逐一分析各种迭代器。
输入迭代器
Input iterators can only step forward element-by-element with read access. Thus, they return values elementwise
输入迭代器的操作:
Table 7.2. Operations of Input Iterators
| Expression | Effect |
|---|---|
| *iter | Provides read access to the actual element |
| iter ->member | Provides read access to a member (if any) of the actual element |
| ++iter | Steps forward (returns new position) |
| iter++ | Steps forward (returns old position) |
| Iter1 == iter2 | Returns whether two iterators are equal |
| Iter1 != iter2 | Returns whether two iterators are not equal |
| TYPE(iter) | Copies iterator (copy constructor) |
注意:输入迭代器只能够读取一次元素。几乎所有其他迭代器都有输入迭代器的功能。
输出迭代器
Output iterators are the counterparts of input iterators. They can only step forward with write access. Thus, you can assign new values only element-by-element. You can't use an output iterator to iterate twice over the same range. The goal is to write a value into a "black hole" (whatever that means). So, if you write something for the second time at the same position into the same black hole, it is not guaranteed that you will overwrite a previous value. Table 7.3 lists the valid operations for output iterators. The only valid use of operator * is on the left side of an assignment statement.
Table 7.3. Operations of Output Iterators
| Expression | Effect |
|---|---|
| *iter = value | Writes value to where the iterator refers |
| ++iter | Steps forward (returns new position) |
| iter++ | Steps forward (returns old position) |
| TYPE (iter) | Copies iterator (copy constructor) |
前向迭代器
前向迭代器是输入迭代器和输出迭代器的组合。它包含输入迭代器的所有功能和输出迭代器的大部分功能。
下面是前向迭代器的操作:
Table 7.4. Operations of Forward Iterators
| Expression | Effect |
|---|---|
| *iter | Provides access to the actual element |
| iter-> member | Provides access to a member of the actual element |
| ++iter | Steps forward (returns new position) |
| iter++ | Steps forward (returns old position) |
| iter1 == iter2 | Returns whether two iterators are equal |
| iter1 != iter2 | Returns whether two iterators are not equal |
| TYPE() | Creates iterator (default constructor) |
| TYPE(iter) | Copies iterator (copy constructor) |
| iter1 = iter2 |
Assigns an iterator |
Unlike input iterators and output iterators, forward iterators can refer to the same element in the same collection and process the same element more than once.
为什么前向迭代器只包含输出迭代器的大部分功能,而不是全部功能呢?
对于输出迭代器,在不检查是不是队列的末尾就写数据是正确的。事实上,是不可以将一个输出迭代器和一个终端迭代器作比较的,因为输出迭代器没有对比的操作。
//OK for output iterators
//ERROR for forward iterators
while (true) {
*pos = foo();
++pos;
}
对于前向迭代器,在访问迭代器指向的数据之前,必须确定该迭代器是否是正确的。这就是为什么上面的循环中的代码对前向迭代器是错误的。如果该迭代器指向的是NULL,那么在没有检查之前便去访问未知地址,会出现未定义行为。
//OK for forward iterators
//IMPOSSIBLE for output iterators
while (pos != coll.end()) {
*pos = foo();
++pos;
}
双向迭代器
Bidirectional iterators are forward iterators that provide the additional ability to iterate backward over the elements. Thus, they provide the decrement operator to step backward (Table 7.5).
Table 7.5. Additional Operations of Bidirectional Iterators
| Expression | Effect |
|---|---|
| -- iter | Steps backward (returns new position) |
| iter-- | Steps backward (returns old position) |
随机访问迭代器
Random access iterators are bidirectional iterators that can perform random access. Thus, they provide operators for "iterator arithmetic" (in accordance with the "pointer arithmetic" of ordinary pointers). That is, they can add and subtract offsets, process differences, and compare iterators with relational operators such as < and >. Table 7.6 lists the additional operations of random access iterators.
Random access iterators are provided by the following objects and types:
Containers with random access (vector, deque)
Strings (string, wstring)
Ordinary arrays (pointers)
Table 7.6. Additional Operations of Random Access Iterators
| Expression | Effect |
|---|---|
| iter[n] | Provides access to the element that has index n |
| iter+=n | Steps n elements forward (or backward, if n is negative) |
| iter-=n | Steps n elements backward (or forward, if n is negative) |
| iter+n | Returns the iterator of the nth next element |
| n+iter | Returns the iterator of the nth next element |
| iter-n | Returns the iterator of the nth previous element |
| iter1-iter2 | Returns the distance between iter1 and iter2 |
| iter1<iter2 | Returns whether iter1 is before iter2 |
| iter1>iter2 | Returns whether iter1 is after iter2 |
| iter1<=iter2 | Returns whether iter1 is not after iter2 |
| iter1>=iter2 | Returns whether iter1 is not before iter2 |
The following program demonstrates the special abilities of random access iterators:
// iter/itercat.cpp #include <vector>
#include <iostream>
using namespace std; int main()
{
vector<int> coll; //insert elements from -3 to 9
for (int i=-; i<=; ++i) {
coll.push_back (i);
} /* print number of elements by processing the distance between beginning and end
* - NOTE: uses operator -for iterators
*/
cout << "number/distance: " << coll.end()-coll.begin() << endl; /* print all elements
* - NOTE: uses operator < instead of operator ! =
*/
vector<int>::iterator pos;
for (pos=coll.begin(); pos<coll.end(); ++pos) {
cout << *pos << ' ';
}
cout << endl; /* print all elements
* - NOTE: uses operator [ ] instead of operator *
*/
for (int i=; i<coll.size(); ++i) {
cout << coll.begin() [i] << ' ';
}
cout << endl; /* print every second element
* - NOTE: uses operator +=
*/
for (pos = coll.begin(); pos < coll.end()-; pos += ) {
cout << *pos << ' ';
}
cout << endl;
}
The output of the program is as follows:
number/distance: 13
-3 -2 -1 0 1 2 3 4 5 6 7 8 9
-3 -2 -1 0 1 2 3 4 5 6 7 8 9
-3 -1 1 3 5 7
STL之Iterator(迭代器)的更多相关文章
- STL之iterator(迭代器)
3.迭代器简单介绍 除了使用下标来訪问vector对象的元素外,标准库还提供了訪问元素的方法:使用迭代器.迭代器是一种检查容器内元素而且遍历元素的数据类型. 百科释义: 迭代器(iterator)是一 ...
- C++ Iterator迭代器介绍及Iterator迭代器用法代码举例
C++ Iterator迭代器介绍 迭代器可被用来访问一个容器类的所包函的全部元素,其行为像一个指针.举一个例子,你可用一个迭代器来实现对vector容器中所含元素的遍历.有这么几种迭代器如下: 迭代 ...
- ES6笔记(6)-- Set、Map结构和Iterator迭代器
系列文章 -- ES6笔记系列 搞ES6的人也是够无聊,把JS弄得越来越像Java.C++,连Iterator迭代器.Set集合.Map结构都出来了,不知道说什么好... 一.简单使用 1. iter ...
- vector容器+iterator迭代器
关于vector容器的详细描述,可参考:http://www.jb51.net/article/41648.htm 关于iterator迭代器的描述,可参考http://www.cppblog.c ...
- 【转】Java学习之Iterator(迭代器)的一般用法 (转)
[转]Java学习之Iterator(迭代器)的一般用法 (转) 迭代器(Iterator) 迭代器是一种设计模式,它是一个对象,它可以遍历并选择序列中的对象,而开发人员不需要了解该序列的底层结构.迭 ...
- 设计模式(十五):Iterator迭代器模式 -- 行为型模式
1.概述 类中的面向对象编程封装应用逻辑.类,就是实例化的对象,每个单独的对象都有一个特定的身份和状态.单独的对象是一种组织代码的有用方法,但通常你会处理一组对象或者集合. 集合不一定是均一的.图形用 ...
- C#:iterator 迭代器/partial class 分布类/泛型
C#:iterator 迭代器/partial class 分布类/泛型 iterator 迭代器 写个最简单的迭代,(迭代一个字符串数组): 1.实现接口中的方法: 1 using System; ...
- [设计模式] Iterator - 迭代器模式:由一份奥利奥早餐联想到的设计模式
Iterator - 迭代器模式 目录 前言 回顾 UML 类图 代码分析 抽象的 UML 类图 思考 前言 这是一包奥利奥(数组),里面藏了很多块奥利奥饼干(数组中的元素),我将它们放在一个碟子上慢 ...
- Python 中 Iterator(迭代器)和Iterable(迭代对象)的区别
直接可以用作for循环的数据类型有以下几种: tuple.list.dict.str等, 上述数据类型可以用作for循环的叫做可迭代对象Iterable.可以使用isinstance判断一个对象是否是 ...
- 使用Iterator迭代器循环集合
1.Iterator迭代器用于遍历集合元素,获取迭代器可以使用. 2.Iterator提供了统一遍历集合元素的 方式 ,其提供了用于遍历集合的连个方法----- boolean hasNext()判 ...
随机推荐
- 关于JPush使用CPU占有率100%的情况
跑模拟器cpu占有率120%+,开始没注意,真机时候手机发烫的厉害,看了下CPU和线程 如图,发现占有率最高的是com.apple.CFSocket.private和org.hxhg.jpush.th ...
- Delphi 用ToolButton和MonthCalendar实现DateTimePicker的功能
效果图如下: 实现平台:xp xe2,其中以上功能的实现,核心主要是参考了万一老师的资料,连接:http://www.cnblogs.com/del/archive/2011/05/12/204411 ...
- openssl 使用非阻塞 bio
序 在项目中需要访问 https 加密的网页,为了保证并发性,需要用到非阻塞的 socket,搜索发现,这种使用场景的相关介绍不是很多,所以这里记录一下使用的过程. 在项目中,所使用的 ssl 库是老 ...
- TalkingData游戏版本在Cocos2d-x 3.0使用
Cocos2dx在3.0的版本中改动确实不少啊,所以导致原来可以在Cocos2.x版本上的demo都不能直接用,所以不得不重要写一个新的demo 但是TalkingData的库一直都是可以用的,只是之 ...
- SCXML和QScxml使用总结
最近接触了SCXML这个状态描述文本,简单来讲就是描述了整个状态的变迁过程的一种XML格式的表格.Qt labs中有一个项目就是QScxml,它基于QStateMachine上层制作,可以直接读取SC ...
- VISUAL STUDIO 2005连接MYSQL数据库
// mysql.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include <string.h> #include &l ...
- Flask 快速入门
最简单的flask程序 from flask import Flask app = Flask(__name__) @app.route('/') def hello_world(): return ...
- Python第三方库(模块)"scikit learn"以及其他库的安装
scikit-learn是一个用于机器学习的 Python 模块. 其主页:http://scikit-learn.org/stable/. GitHub地址: https://github.com/ ...
- 探究Android中Listview显示错乱问题
问题 最近在项目中遇到过一个很棘手的问题,就是ListView在滑动后就莫名其妙的显示错乱,网上查阅资料后问题很容易的就解决了,但是对于问题产生的原因仍是一知半解,所以不甘心的我定下心来,狠读源码,终 ...
- Linux下设置静态IP和获取动态IP的方法
Linux下为机器设置静态IP地址: vim /etc/sysconfig/network-scripts/ifcfg-eth0 修改这个文件内容如下形式: # Intel Corporation ...