Сначала давайте немного обобщим:
typedef int value_type;
typedef std::vector<value_type*> inner_range;
typedef std::vector<inner_range*> outer_range;
Теперь итератор:
struct my_iterator : std::iterator_traits<inner_range::iterator>
{
typedef std::forward_iterator_tag iterator_category;
my_iterator(outer_range::iterator const & outer_iterator,
outer_range::iterator const & outer_end)
: outer_iterator(outer_iterator), outer_end(outer_end)
{
update();
}
my_iterator & operator++()
{
++inner_iterator;
if(inner_iterator == inner_end)
{
++outer_iterator;
update();
}
return *this;
}
reference operator*() const
{
return *inner_iterator;
}
bool operator==(my_iterator const & rhs) const
{
bool lhs_end = outer_iterator == outer_end;
bool rhs_end = rhs.outer_iterator == rhs.outer_end;
if(lhs_end && rhs_end)
return true;
if(lhs_end != rhs_end)
return false;
return outer_iterator == rhs.outer_iterator
&& inner_iterator == rhs.inner_iterator;
}
private:
outer_range::iterator outer_iterator, outer_end;
inner_range::iterator inner_iterator, inner_end;
void update()
{
while(outer_iterator != outer_end)
{
inner_iterator = (*outer_iterator)->begin();
inner_end = (*outer_iterator)->end();
if(inner_iterator == inner_end)
++outer_iterator;
else
break;
}
}
};
Этот класс предполагает, что внешние итераторы содержат указатели на внутренние диапазоны, что было требованием в вашем вопросе. Это отражено в элементе update
, в стрелках перед begin()
и end()
. Вы можете заменить эти стрелки точками, если хотите использовать этот класс в более распространенной ситуации, когда внешний итератор содержит внутренние диапазоны по значению. Обратите внимание, что этот класс не зависит от того, что внутренний диапазон содержит указатели, это должны знать только клиенты этого класса.
Код может быть короче, если мы используем boost::iterator_facade
, но нет необходимости добавлять буст-зависимость для чего-то такого простого. Кроме того, единственными сложными частями являются операции равенства и приращения, и мы все равно должны их кодировать.
Я оставил в качестве «упражнений для читателя» следующие элементы котельной:
- Постфиксный итератор приращения
- оператор! =
- конструктор по умолчанию
- оператор->
Еще одно интересное упражнение - превратить его в шаблон, который работает с произвольными контейнерами. Код в основном такой же, за исключением того, что вы должны добавить аннотации typename
в нескольких местах.
Пример использования:
int main()
{
outer_type outer;
int a = 0, b = 1, c = 2;
inner_type inner1, inner2;
inner1.push_back(&a);
inner1.push_back(&b);
inner2.push_back(&c);
outer.push_back(&inner1);
outer.push_back(&inner2);
my_iterator it(outer.begin(), outer.end());
e(outer.end(), outer.end());
for(; it != e; ++it)
std::cout << **it << "\n";
}
Какие отпечатки:
0 1 2