zoukankan      html  css  js  c++  java
  • C++基础之迭代器

    迭代器的分类

    插入迭代器(insert iterator):绑定一个容器上后可以向容器中插入元素;

    流迭代器(stream iterator):绑定在输入输出流中,可以遍历关联的流;

    反向迭代器(reverse iterator):迭代器向后移动,标准库容器中除了forward_list外都有反向迭代器;

    移动迭代器(move iterator):使用该迭代器移动其中元素;

    插入迭代器(insert iterator)

    back_inserter:创建一个使用push_back的迭代器;

    front_inserter:创建一个使用push_front的迭代器;

    inserter:创建一个使用insert的迭代器;

    注意:只有容器支持push_front的情况下,才能使用front_inserter;只有容器支持push_back的情况下,才能使用back_inserter;

    若it是inserter生成的插入迭代器,则*it = val;等价于it = c.insert(it,val);++it;

    顾名思义back_inserter始终将元素插入到末尾,front_inserter始终将元素插入到头部。

    list<int> lst1 = {1,2,3,4};
    list<int> lst2,lst3;
    
    //复制lst1到lst2,每次将lst1中的元素复制插入到lst2的前面
    copy(lst1.cbegin(),lst1.cend(),front_inserter(lst2));
    //复制lst1到lst3,每次将lst1中的元素复制插入到lst3的前面
    copy(lst1.cbegin(),lst1.cend(),inserter(lst3,lst3.begin()));

    流迭代器(stream iterator)

    虽然iostream类型不是容器,但是标准库定义用于IO类型对象的迭代器。istream_iterator读取输入流,ostream_iterator向输出流写数据。

    istream_iterator操作

    可以对任何定义了输入运算符(>>运算符)的类型定义istream_iterator。

    从标准输入读取数据存入到数组中:

    istream_iterator<int> in_iter(cin);//从cin中读取数据
    istream_iterator<int> eof;//尾后迭代器
    vector<int> vec(in_iter, eof);//将输入的数据存入数组中

    使用算法将输入数据求和:

    istream_iterator<int> in_sum(cin), eof;//从cin中读取数据,尾后迭代器
    cout << accumulate(in_sum, eof, 0) << endl;//将输入的数据存入数组中

    istream_iterator允许使用懒惰求值

    当我们将istream_iterator绑定到一个流上时,并不保证迭代器立即从流中读取数据;即具体实现中可以推迟从流中读取数据。

    标准库保证的是在我们第一次解引用迭代器之前,已经完成从流中读取数据的操作。

    ostream_iterator操作

    可以对任何定义了输出运算符(<<运算符)的类型定义ostream_iterator。

    ostream_iterator<T> out(os);//out将类型为T的值写到输出流os中
    ostream_iterator<T> out(os,d);//out将类型为T的值写到输出流os中,每个值后面输出一个C风格的字符串d(一个字符串字面常量或一个指向空字符结尾的字符数组指针)

    不允许空的或尾后的ostream_iterator

    ostream_iterator<int> out_iter(cout," ");
    copy(vec.cbegin(),vec.cend(),out_iter));//输出数组vec中的所有元素,用空格隔开
    cout << endl;

    反向迭代器(reverse iterator)

    反向迭代器(reverse iterator)从容器的尾元素向首元素移动的迭代器,此时递增表示移动到前一个元素,递减表示移动到后一个元素,移动到第一个元素的前一个位置表示结束。

    除了forward_list之外的容器都有反向迭代器,可以通过rbegin、rend、crbegin、crend来获得反向迭代器。可以看出,它也有const和非const两个版本。

    反向输出数组的所有元素:

    vector<int> arr = {0,1,2,3,4,5,6,7,8,9};
    for(auto riter = arr.crbegin();riter != arr.crend();++riter)
        cout << *riter << endl;

    降序排序的另一种写法:

    sort(vec.rbegin(),vec.rend());

    注意流迭代器不支持递减运算,因为不能在一个流中反向移动。

    反向迭代器转换为普通的迭代器

    string line = "FIRST,MIDDLE,LAST";
    auto pos = find(line.crbegin(), line.crend(), ',');//找到最后一个单词
    cout << string(line.crbegin(), pos) << endl;//输出TSAL
    cout << string(pos.base(), line.cend()) << endl;//输出LAST

    上面通过调用reverse_iterator的base成员函数来完成反向迭代器向普通的迭代器的转换。

    注意这两者的转换,关键在于[line.crbegin(), pos)和[pos.base(), line.cend())指向line中相同的元素范围,为了实现这个pos和pos.base()必须指向相邻位置而不是相同位置。

    移动迭代器(move iterator)

    泛型算法对应的5中迭代器操作

    输入迭代器    只读,不写;单遍扫描,只能递增

    输出迭代器    只读,不写;单遍扫描,只能递增

    向前迭代器    可读写;多遍扫描,只能递增

    双向迭代器    可读写;多遍扫描,可递增递减

    随机访问迭代器  可读写;多遍扫描,支持迭代器所有运算

    C++标准指明了泛型和数值算法的每个迭代器参数的最小类别。

    例如find算法要求对序列一遍扫描,对元素只读操作,因此至少需要输入迭代器;replace函数需要一对迭代器,至少是向前迭代器;replace_copy的前两个迭代器至少是向前迭代器,第三个迭代器表示目前位置,必须至少是输出迭代器。

    输入迭代器(input iterator)要支持:

    它只用于顺序访问,对于输入迭代器,*it++保证有效,但是,递增他可能导致其他指向流的迭代器失效,因此只能用于单遍扫描算法,例如find和accumulate。

    1. 比较两个迭代器的相等和不相等(==、!=)
    2. 迭代器的前置和后置递增运算(++)
    3. 读取元素的解引用运算(*)
    4. 箭头运算符(->)等价于解引用

    输出迭代器(output iterator)要支持:

    只能向输出迭代器赋值一次,且它只能用于单遍扫描算法,用作目的位置的迭代器通常是输出迭代器。例如copy的第三个迭代器。

    1. 迭代器的前置和后置递增运算(++)
    2. 读取元素的解引用运算(*)

    向前迭代器(forward iterator)要支持:

    可以读写元素,只能在序列中沿一个方向移动,支持所有输入迭代器和输出迭代器的操作,可以多次读写同一个元素;因此可以保存前向迭代器的状态,可以对序列多次扫描。

    双向迭代器(bidirectional iterator)要支持:

    可以读写元素,只能在序列中正反两个方向移动,支持所有前向迭代器的操作,还支持前置和后置递减运算符。例如reverse要求双向迭代器。

    随机访问迭代器(random-access iterator)

    提供在常量时间内访问序列中任意元素,支持双向迭代器的所有功能。

    1. 比较两个迭代器相对位置的关系运算符(<、>、==、!=、...)
    2. 迭代器和整数的加减运算符(++、——、+=、—=)
    3. 两个迭代器减法运算
    4. 下标运算符,和*等价
  • 相关阅读:
    [THREEJS]坐标高精度问题
    纹理的寻址方式
    [1009]JS语言精髓与编程实践笔记1
    万向锁
    (转)primitive restarting
    西藏游记
    平衡二叉树(AVL tree)
    初探redis事务
    redis发布与订阅
    初探redis分布式锁
  • 原文地址:https://www.cnblogs.com/yeqluofwupheng/p/6799185.html
Copyright © 2011-2022 走看看