7

在网上搜索了很多,但找不到问题的答案。

我正在使用它的 reverse_iterator 向 std::list 插入一个值。虽然插入按预期发生在适当的位置,但我注意到用于插入的原始 reverse_iterator 的值发生了变化。完全不相关的 reverse_iterator 的值也会发生变化。我已经能够在一个简单的例子中重现这个

#include <iostream>
#include <list>
#include <string>

int main()
{
    // Creating a list of values 1, 2, 4, 5
    std::list<int> myList;
    myList.push_back(1);
    myList.push_back(2);
    myList.push_back(4);
    myList.push_back(5);

    // Changing it to 1, 2, 3, 4, 5 by inserting 3
    std::list<int>::reverse_iterator revIter = myList.rbegin();
    while(2 != *revIter)
    {
        std::cout << *revIter << "\t";
        ++revIter;
    }

    std::cout << "\n" << "Reverse iterator now points to " << *revIter;
    // Creating a copy of the reverse Iter before inserting.
    std::list<int>::reverse_iterator newRevIter = revIter;
    myList.insert(revIter.base(), 3);

    // Checking the values of revIter and newRevIter 
    std::cout << "\n" << "Reverse Iterator now points to " << *revIter; // UNEXPECTED RESULT HERE
    std::cout << "\n" << "New Reverse Iterator now points to " << *newRevIter; // UNEXPRECTED RESULT HERE

    std::cout << "\n" << "Printing final list:" << "\n";
    for(std::list<int>::iterator iter = myList.begin(); myList.end() != iter; ++iter)
    {
        std::cout << *iter << "\t"; // Results are fine
    }

    return 0;
}

结果

5    4
Reverse iterator now points to 2
Reverse iterator now points to 3
New Reverse iterator now points to 3
Printing final list:
1    2    3    4    5

这是预期的行为。如果是这样,如何使用反向迭代器将项目插入列表(或者在这方面它没用)?

4

2 回答 2

4

我会避免使用反向迭代器(一般来说,特别是对于除顺序横向之外的任何东西)。正向和反向迭代器的工作方式不同,在正向迭代器进入列表的情况下,迭代器跟踪您通过 访问的节点operator*,但在反向情况下,迭代器跟踪列表中的下一个元素。取消引用反向迭代器的行为获取迭代器引用的节点的前驱并从中提取值。图形化(f 是正向迭代器,r 是反向迭代器)

  f
1 2 4
    r

正向迭代器 f 和反向迭代器 r 在取消引用时都会产生 2,但它们跟踪的节点不同。当您使用 r 插入时,您在 2 和 4 之间插入,但底层迭代器仍然指向持有 4 的节点:

  f
1 2 3 4
      r

现在,如果您取消引用 r,则适用与上述相同的过程。获取当前节点的前任,并打印值,只是现在 4 的前任是 3,这就是你得到的。

这是预期的行为。如果是这样,如何使用反向迭代器将项目插入列表(或者在这方面它没用)?

是的,这是预期的行为。如何使用反向迭代器将项目插入列表?了解它是如何工作的。

于 2013-11-19T14:46:09.793 回答
0

插入后的不变量是std::reverse_iterator<>::base(),而不是std::reverse_iterator<>本身。但是,base()与以下相比,它确实针对先前的元素reverse_iterator

http://en.cppreference.com/w/cpp/iterator/reverse_iterator

让我有点困扰的是,当我不得不std::distancebegin()rend().base()):

  std::cout << "\n" 
        << "Reverse Iterator now points to " 
        << *revIter << "-" << *(revIter.base())<< "-" 
        << std::distance(revIter.base(), myList.rend().base()); 

我有:

反向迭代器现在指向 2-4-3

反向迭代器现在指向 3-4-3

base()或者我希望第二个是“3-4-4”,因为元素是在...之前插入的

于 2013-11-19T14:22:17.747 回答