for each
оператор в VC ++ при использовании в неуправляемом классе:
for each (T x in xs)
{
...
}
это просто синтаксический сахар для этого:
for (auto iter = xs.begin(), end = xs.end(); iter != end; ++iter)
{
T x = *iter;
}
Где auto
означает, что тип переменной выводится автоматически из типа инициализатора.
Другими словами, вам нужно предоставить begin()
и end()
методы для вашего класса, которые возвращали бы начальный и конечный итераторы ввода для него.
Вот пример класса, который переносит istream
и позволяет вам перебирать все строки в нем:
#include <istream>
#include <iostream>
#include <fstream>
#include <string>
class lines
{
public:
class line_iterator
{
public:
line_iterator() : in(0)
{
}
line_iterator(std::istream& in) : in(&in)
{
++*this;
}
line_iterator& operator++ ()
{
getline(*in, line);
return *this;
}
line_iterator operator++ (int)
{
line_iterator result = *this;
++*this;
return result;
}
const std::string& operator* () const
{
return line;
}
const std::string& operator-> () const
{
return line;
}
friend bool operator== (const line_iterator& lhs, const line_iterator& rhs)
{
return (lhs.in == rhs.in) ||
(lhs.in == 0 && rhs.in->eof()) ||
(rhs.in == 0 && lhs.in->eof());
}
friend bool operator!= (const line_iterator& lhs, const line_iterator& rhs)
{
return !(lhs == rhs);
}
private:
std::istream* const in;
std::string line;
};
lines(std::istream& in) : in(in)
{
}
line_iterator begin() const
{
return line_iterator(in);
}
line_iterator end() const
{
return line_iterator();
}
private:
std::istream& in;
};
int main()
{
std::ifstream f(__FILE__);
for each (std::string line in lines(f))
{
std::cout << line << std::endl;
}
}
Обратите внимание, что реализация line_iterator
на самом деле несколько больше, чем минимум, необходимый для for each
; однако это минимальная реализация, которая соответствует требованиям входных итераторов, и, таким образом, этот класс также может использоваться со всеми алгоритмами STL, которые работают с входными итераторами, такими как std::for_each
, std::find
и т. д.