c++ - 这是制作迭代器的可接受方式吗?

标签 c++ c++11 for-loop iterator operator-overloading

我喜欢 C++ 中的范围 for 循环,并且想像这样使用它:

#include <bits/stdc++.h>

int main()
{
    for (auto s : LineReader("my-very-big-textfile.txt")) {
        cout << s << endl;
    }
    return 0;
}

此处的目的是循环访问一些数据(无需先将所有数据读入容器)。在这种情况下,文本字符串是文本文件中的行。但通常它可以是任何东西(包括生成的数据)。

这里 LineReader 返回一个可迭代的“伪”容器。为了让它工作,for 循环需要来自 LineReader 对象的迭代器。在 C++ 中,范围是根据开始和结束迭代器定义的。但是我想使用 range for-loop 来遍历开始时可能不知道结尾的数据(例如,在(过大的)文本文件中逐行读取而不先通过它来找到结尾。)。

所以我这样定义:

免责声明:显示原理的示例代码,因此我不会过度使用 std::、错误处理、私有(private)/公共(public)关键字等来“纠缠”它...

struct ReadLineIterator {
    ifstream ifs;
    string line;

    ReadLineIterator() { }
    ReadLineIterator(string filename) : ifs(filename) { }

    bool operator!=(ReadLineIterator& other) {
        return !ifs.eof();
    }

    ReadLineIterator& operator++() {
        getline(ifs, line, '\n');
        return *this;
    }
    string operator*() {
        return line;
    }
};

struct LineReader
{
    string filename;
    LineReader(const string& filename) : filename(filename) { }

    ReadLineIterator begin()
    {
       return ReadLineIterator(filename);
    }

    ReadLineIterator end() // return a not used dummy iterator since this method must exist
    {
        return ReadLineIterator();
    }
};

当我运行它时,它起作用了。但我怀疑是否

bool operator!=(ReadLineIterator& other) {
    return !ifs.eof();
}

是使此运算符检测序列结尾的正确方法。这是因为我没有任何合适的结束对象(end() 方法只返回一个虚拟迭代器)并且也没有对它进行比较。相反,我检查流是否为空。

但我不知道我怎么能以任何其他方式做到这一点?现在我对这种方法很满意,因为它对我有用,但如果知道是否有更好的方法来做同样的事情,那就太好了。另外,很高兴知道这是否适用于所有 (C++) 编译器(我使用的是 GCC),如果适用,它是否适用于 future 的 C++ 标准,在这些标准中迭代器的处理方式可能会有所不同。

最佳答案

我会分两部分来做。

一个是range仅充当流迭代器包装器的类:

template <class T>
class istream_range {
    std::istream_iterator<T> b;
    std::istream_iterator<T> e;
public:
    istream_range(std::istream &is)
        : b(std::istream_iterator<T>(is))
        , e(std::istream_iterator<T>())
    {}

    std::istream_iterator<T> begin() { return b; }
    std::istream_iterator<T> end() { return e; }
};

所以这让我们使用 istream_iterator s 在基于范围的 for 循环中:

for (auto const &s : istream_range<foo>(myfile))
    // do something with s

istream_iterator使用 operator>>从指定的文件中提取项目,所以第二部分只是一个提取一行的小类型:

class line {
    std::string data;
public:
    friend std::istream &operator>>(std::istream &is, line &l) {
        std::getline(is, l.data);
        return is;
    }
    operator std::string() const { return data; }    
};

所以,有了这个我们的 for循环变成这样:

for (auto const &s : istream_range<line>(myfile))
    // do something with s

这样做的明显优势是将两者解耦:我们可以使用 istream_range<T>处理 T 的文件, 对于任何 T正常的流提取做了“正确的事情”(包括我们目前无法感知的许多自定义提取器)。

previous question 的答案中涵盖了更多的可能性。 (包括一个 LineInputIterator,它似乎更接近您的要求)。

关于c++ - 这是制作迭代器的可接受方式吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44812236/

相关文章:

c++ - 如何开启英特尔C++编译器12.1.2的C++0x

for-loop - Lua 中的pairs() 和ipairs() 有什么区别?

c++ - 如何将枚举类用作一组标志?

c++ - 使用 Linux 访问并口

c++ - 有没有一种方法可以在不创建该类实例的情况下获取 QObject 派生类的类名?

c++ - std::thread::detach 在原始调用者被销毁后导致崩溃

c++ - 从整型常量表达式转换为空指针

c++ - 如何使用 Visual C++ 中的 Delphi 寄存器调用约定调用函数?

c - 如何增加我想要操作的数组 block

java - For 循环计数不正确,以某种方式从错误的变量计数?