请原谅我,如果这是一个微不足道的问题,我只是在学习C ++并尝试围绕某些概念。特别是在谈到迭代器时,我完全迷失了。
假设我有一个表示某种数据结构的自定义类,其中一个成员是一个整数向量。我想为该类编写一个双向迭代器,它只输出向量中的偶数。有一种简单而有教育意义的方式吗?我不想使用STL以外的库。
答案 0 :(得分:3)
不确定自己的迭代器是否容易。但最好的方法是使用条件for_each
函数。
std::for_each
对每个元素执行操作。在某些特定元素上创建for_each_if
操作非常容易。例如,下面的程序只打印向量中的偶数(4,6和8)。
#include <iostream>
#include <vector>
using namespace std;
struct is_even {
typedef bool return_type;
bool operator() (const int& value) {return (value%2)==0; }
};
struct doprint {
bool operator() (const int& value) { std::cout << value << std::endl; }
};
template <class InputIterator, class Predicate, class Function>
void for_each_if(InputIterator first, InputIterator last, Function f, Predicate pred)
{
while ( first != last )
{
if (pred (*first))
f(*first++);
else
first ++;
}
}
int main()
{
std::vector<int> v;
v.push_back( 4 );
v.push_back( 5 );
v.push_back( 6 );
v.push_back( 8 );
for_each_if( v.begin(), v.end(), doprint(), is_even());
return 0;
}
答案 1 :(得分:3)
根据您的要求,从vector::iterator
派生可能是最简单的:
class my_iterator : private std::vector<int>::iterator {
typedef std::vector<int>::iterator base;
base m_base_end; // Stores the real end iterator of the internal vector.
// We need it so that the even/odd checking code
// doesn't run off the end of the vector
// (initialize it in a constructor, omitted here for
// brevity).
public:
using base::operator*;
using base::operator->;
// etc. for other members...
// except for increment:
my_iterator& operator++()
{
do {
base::operator++();
} while( static_cast<base&>(*this) != m_base_end
&& base::operator*() % 2 );
return *this;
}
my_iterator operator++(int)
{
my_iterator temp;
operator++();
return temp;
}
// TODO: decrement and other arithmetic operators
};
它仍然是相当多的样板,你也想对const_iterator
做同样的事情(我可能会让上面的类成为一个模板,让它变得更容易)。
考虑一下Boost - 它只有filter_iterator
才能达到此目的。如果这不适合你,还有iterator_adaptor
。
答案 2 :(得分:1)
#include <vector>
#include <iostream>
class X {
public:
class EvenIterator {
public:
EvenIterator(std::vector<int>::iterator it, std::vector<int>::iterator end) : it(it), end(end) {
while (true) {
if (isEven(*it)) {
break;
} else if (it == end) {
break;
}
it++;
}
}
bool operator != (const EvenIterator& evenIt) {
return evenIt.it != this->it;
}
int operator * () {
return *it;
}
EvenIterator operator ++ () {
while (true) {
it++;
if (isEven(*it)) {
return EvenIterator(it, end);
} else if (it == end) {
return EvenIterator(it, end);
}
}
}
private:
std::vector<int>::iterator it;
std::vector<int>::iterator end;
};
static bool isEven(int number) {
return number % 2 == 0;
}
void add(int number) {
v.push_back(number);
}
EvenIterator evenBegin() {
return EvenIterator(v.begin(), v.end());
}
EvenIterator evenEnd() {
return EvenIterator(v.end(), v.end());
}
private:
std::vector<int> v;
};
int main() {
X x;
x.add(1);
x.add(2);
x.add(3);
x.add(2);
x.add(2);
x.add(31);
x.add(56);
x.add(101);
for (X::EvenIterator it = x.evenBegin(); it != x.evenEnd(); ++it){
std::cout << *it << std::endl; // only prints the even numbers
}
}