除了为每个容器定义的迭代器之外,标准库在头文件<iterator>还定义了额外集中迭代器, 包括:

  • 插入迭代器,这些迭代器被绑定到一个容器上,可以向容器插入元素。
  • 流迭代器,    这些迭代器被绑定到输入或输出流上,可以遍历所关联的IO流
  • 反向迭代器,这些迭代器向后而不是向前移动,除了forward_list之外的标准库容器都有反向迭代器。
  • 移动迭代器,这些专用得对得起不是拷贝而是移动其中的元素。

插入迭代器

插入迭代器操作:
it = t   在it指定的位置插入值t
*it, ++it, --it

有三种插入迭代器:

  • back_inserter 创建一个使用push_back的迭代器。
  • front_inserter创建一个使用push_front的迭代器。
  • inserter创建一个使用insert的迭代器。此函数接受第二个参数,这个参数必须是一个指向给定容器的迭代器。元素被插入到指定的迭代器所表示的元素之前。
只有容器支持push_back才可以使用back_inserter, 同理,只有容器支持push_front时才可以使用front_inserter
当调用inserter时,得到一个迭代器,接下来使用它时,会将元素插入到这个迭代器所指向的元素之前的位置。即,如果it是由inserter生成的迭代器,则下面这样的赋值语句
*it = val;

效果和下面的代码一样:

it = c.insert(it, val);
++it; // 递增it使它指向原来的元素

而front_inserter生成迭代器的行为和inserter不一样,front_inserter会将传入的元素逆序插入到容器内:

list<int> lst1 = {1, 2, 3, 4};
list<int> lst2, lst3;
copy(lst.begin(), lst.end(), front_inserter(lst2));
copy(lst.begin(), lst.end(), inserter(lst3, llst3.begin()));

我们如果输出lst2和lst3的内容,则可以看到lst2 = 4, 3, 2, 1  而lst3 = 1, 2, 3, 4

iostream迭代器

标准库为iostream定义了可用于这些IO类型对象的迭代器。 istream_iterator读取输入流, ostream_iterator向一个输出流写数据。

1. istream_iterator操作

当创建一个流迭代器时,必须指定迭代器将要读写的对象类型。一个istream_iterator使用 >> 来读取流。因此,istream_iterator要读取的类型必须定义了 >> 运算符。
创建一个istream_iterator时,我们可以将它绑定到一个流。
// 从cin读取string
istream_iterator<string> isiter(cin); // 从 file读取string
ifstream ifs("file");
istream_iterator<string> isiter1(ifs);

我们还可以默认初始化一个流迭代器,这样就创建了一个尅当做尾后值使用的迭代器。

下面是用从标准输入输入数据,存入一个vector:

vector<int> vec;
istream_iterator<int> in_iter(cin);
istream_iterator<int> end; // istream_iterator尾后迭代器
while(in_iter != end)
vec.push_back(*in_iter++);
istream_iterator允许懒惰求值, 当我们将一个istream_iterator绑定到一个流时,标准库并不保证我们立即从流中读取数据,我们可以在使用的时候再读。标准库的实现保证的是,在我们第一次解引用迭代器之前,从流中读取数据的操作已经完成, 立即读取还是推迟读取没什么区别。

2.ostream_iterator操作

同样的, 我们可以对任何定义了 << 运算符的类型定义ostream_iterator。当我们创建了一个ostream_iterator时,我们有可选的第二个参数,它必须是一个C风格字符串, 在输出每个元素后都会打印此字符。必须将ostream_iterator绑定到一个指定的流,不允许空的或ostream_iterator来表示尾后迭代器。
我们可以用ostream_iterator来输出值的序列:
ostream_iterator<int> out(cout, " ");
for (auto const e : vec)
*out ++ = e;
cout << endl;

这个程序会将vec中元素输出,而且元素之间都用" " 隔开。

反向迭代器

反向迭代器就是从容器的尾元素开始向首元素反向移动的迭代器。对于反向迭代器, 递增以及递减的操作会和正常迭代器产生相反的效果。递增一个反向迭代器(++it)会移动到前一个元素, 递减一个反向迭代器(--it),会移动到下一个元素。
除了forward_list之外,其他容器都支持反向迭代器。我们可以调用rbegin、rend、crbegin、crend来获得反向迭代器,这些成员函数返回指向容器尾元素和首元素之前的一个位置的迭代器。例如,逆序打印出vector中的元素:
vector<int> vec{0, 1, 2, 3, 4, 5};
for (auto riter = vec.rbegin(); riter != vec.rend(), ++riter)
cout << *riter << endl;

我们还可以将一个容器的元素从递增排序变为递减顺序:

// 递增顺序
sort(vec.begin(), vec.end());
// 递减顺序
sort(vec.rbegin(), vec.rend());

从一个string中打印第一个单词,假定string的单词之间都用逗号隔开:

string line{"FIRST", "MIDDLE", "LAST"};
auto comma = find(line.begin(), line.end(), ",");
cout << string(line.cbegin(), comma);

如果希望打印最后一个单词,我们可以使用反向迭代器:

auto rcomma = find(line.rbegin(), line.rend(), ",");

而我们试图打印最后一个单词时,有意思的事发生了:
cout << string(line.crbegin(), rcomma); 

这时候会输出 : TSAL
因为我们使用的是反向迭代器,所以会反向处理string,如果我们希望按正常顺序打印string,我们通过调用reverse_iterator的base成员来完成这一转换,此成员函数会返回其对应的普通迭代器:
cout << string(rcomma.base(), line.cend()); 

rcomma和rcomma.base()必须生成相邻的位置而不是同一个位置。crbegin()和cend()也是如此。它们表示的是元素的范围。
反向迭代器的目的是表示元素范围,而这些范围是不对称的。这导致一个重要的结果:当我们从一个普通迭代器初始化一个反向迭代器, 或是给一个反向迭代器赋值时,结果迭代器与原迭代器指向的并不是相同的元素。
※ 算法命名规范
接受谓词参数来代替 <或 <=运算符的算法,以及那些不接受额外参数的算法,通常都是重载的函数。
1. _if版本的算法 
	find(beg, end, val);	  // 查找val第一次出现的位置
find_if(beg, end, pred); // 查找第一个使得谓词pred为真的元素
2. _copy版本的算法
	reverse(beg, end);		// 反转范围中的元素的顺序
reverse(beg, end, dest); // 将元素按逆序拷贝到dest

一些算法溶蚀提供_if和_copy版本:
//从v中删除奇数元素
remove_if(v.begin(), v.end(), [] (int i) { return i % 2; });
// 将偶数元素从v拷贝到v2,v不变。
remove_copy_if(v.begin(), v.end(), back_inserter(v2), [] (int i) { return i % 2;});
特定容器算法
list和forward_list定义了几个成员函数形式的算法,它们定义了独有的sort、merge、remove、reverse和unique。通用版本的sort要求随机访问迭代器,因此不能用于list和forward_list, 因为这两个类型分别提供双向迭代器和前向迭代器。
lst.merge(lst2);            将来自lst2的元素合并入lst,lst和lst2都必须是有序的
lst.merge(lst2, comp); 元素将从lst2中删除,在合并之后,lst2变为空,第一个版本使用 < 运算符, 第二个版本使用给定的比较操作 lst.remove(val); 调用erase删除掉与给定值相等或令一元谓词为真的每个元素
lst.remove_if(pred);
lst.reverse(); 反转lst中元素的顺序
lst.sort(); 使用< 或给定比较操作排序元素
lst.sort(comp);
lst.unique(); 调用erase删除用一个值的连续拷贝,第一个版本使用==, 第二个版本使用给定的二元谓词。
lst.unique(pred);

链表类型还定义了splice成员算法

// lst.splice(args)或flst.splice_after(args)
(p, lst2) p是一个指向lst中元素的迭代器,或一个指向flst首前位置的迭代器。
函数将lst2所有元素移动到lst中p之前的位置或是flst中p之后的位置。将元素从lst2中删除,lst2的类型必须与lst或flst相同,且不能是同一个链表 (p, lst2, p2) p2是一个指向lst2中位置的有效迭代器,将p2指向的元素移动到lst中,或将p2之后的元素移动到flst中,lst2可以使与lst或flst相同的链表 (p, lst2, b, e) b和e必须表示lst2中的合法范围,将给定范围中的元素从lst2移动到lst或flst。lst2与lst(或flst)可以是相同的链表,但p不能指向给定范围中的元素
merge和splice会销毁其参数, 通用版本的merge将合并的序列写到一个给定的目的迭代器,两个输入序列式不变的。而链表版本的merge会销毁给定的链表——元素从参数指定的链表中删除,被合并的调用merge的链表对象中,在merge之后,来自两个链表的元素仍然存在,但它们都已在同一个链表中。

C++ Primer :第十章 :泛型算法之再探迭代器以及其他算法的更多相关文章

  1. C++ Primer : 第十章 : 泛型算法 之 只读、写和排序算法

    大多数算法都定义在<algorithm>头文件里,而标准库还在头文件<numeric>里定义了一组数值泛型算法,比如accumulate. ●  find算法,算法接受一对迭代 ...

  2. C++ Primer : 第十章 : 泛型算法 之 lambda表达式和bind函数

    一.lambda表达式 lambda表达式原型: [capture list] (parameter list) -> retrue type { function body } 一个lambd ...

  3. 【C++ Primer 第10章】再探迭代器

    反向迭代器 • 反向迭代器就是在容器中从尾元素向首元素反向移动的迭代器.对于反向迭代器,递增(以及递减)操作的含义会颠倒过来. • 递增一个反向迭代器(++it)会移动到前一个元素:递减一迭代器(-- ...

  4. 【C++ Primer | 10】再探迭代器

    插入迭代器 1. 测试代码: #include<iostream> #include<vector> #include<list> #include<iter ...

  5. 10.4 再探迭代器-插入/IO/反向

    10.4.1 插入迭代器 插入迭代器接受一个容器,生成一个迭代器,通过向该迭代器赋值可以实现向容器添加元素 (1)back_inserter: 接受一个参数, 示例: auto iter = back ...

  6. 【C++ Primer | 10】泛型算法

    #include<iostream> #include<algorithm> #include<vector> #include<string> #in ...

  7. C++ Primer 学习笔记_43_STL实践与分析(17)--再谈迭代器【中】

    STL实践与分析 --再谈迭代器[中] 二.iostream迭代[续] 3.ostream_iterator对象和ostream_iterator对象的使用 能够使用ostream_iterator对 ...

  8. C++ Primer 学习笔记_44_STL实践与分析(18)--再谈迭代器【下】

    STL实践与分析 --再谈迭代器[下] 三.反向迭代器[续:习题] //P355 习题11.19 int main() { vector<int> iVec; for (vector< ...

  9. C++Primer 第十章

    //1.标准库算法不仅可以应用于容器,还可以应用于内置数组,指针. //2.大多数算法都定义在头文件algorithm中.标准库还在头文件numeric中定义了一组数值泛型算法. //3.算法本身不会 ...

随机推荐

  1. 虚拟机的apache服务器不能被主机访问的问题

    我在centos虚拟机上安装了elasticsearch服务,虚拟机里测试正常,但主机却无法访问elasticsearch.要说的是,虚拟机采用桥接模式,与主机相互ping得通. 后来查了资料发现,这 ...

  2. 精通JS 笔记

    一,javascript数据类型:undefined,null,boolean,number,string,object 五种加一种复杂类型. 注意大小写,区分大不写函数:functiontypeof ...

  3. C语言:文件操作

    以附加方式打开文件,输入数据,关闭文件. #include<stdio.h> #include<stdlib.h> int main() { FILE *fp = NULL; ...

  4. bzoj 3687 bitset的运用

    题目大意: 小呆开始研究集合论了,他提出了关于一个数集四个问题:1. 子集的异或和的算术和.2. 子集的异或和的异或和.3. 子集的算术和的算术和.4. 子集的算术和的异或和.目前为止,小呆已经解决了 ...

  5. SVN Unable to connect to a repository at URL

    方法一:右键菜单的“TortoiseSVN”->“Settings”->“Save Data”对话框中,点击“Authentication data”旁的“Clear”按钮,清除登录凭证. ...

  6. 【转发】linux yum命令详解

    linux yum命令详解 yum(全 称为 Yellow dog Updater, Modified)是一个在Fedora和RedHat以及SUSE中的Shell前端软件包管理器.基於RPM包管理, ...

  7. 重拾java系列一java基础(4)

    本章主要回顾一些类的相关知识: (1)static: static 静态的: 属于类的资源, 使用类名访问.  静态属性: 只有一份的变量  静态方法: 是属于类方法, 可以使用类名直接访问. 静态方 ...

  8. Chapter 1: A Simple Web Server

    这算是一篇读书笔记,留着以后复习看看. Web Server又称为Http Server,因为它使用HTTP协议和客户端(一般是各种各样的浏览器)进行通信. 什么是HTTP协议呢? HTTP协议是基于 ...

  9. 网络图片下载缓存库SDWebImage的使用

    SDWebImage导入问题 最新的SDWebImage由于是基于ARC模式写的,如果创建的是非ARC醒目的童鞋们注意,导入文件夹之后,先添加ImageIO.framework,mapKit.fram ...

  10. 算法导论----VLSI芯片测试; n个手机中过半是好的,找出哪些是好手机

    对于分治(Divide and Conquer)的题目,最重要是 1.如何将原问题分解为若干个子问题, 2.子问题中是所有的都需要求解,还是选择一部分子问题即可. 还有一点其实非常关键,但是往往会被忽 ...