Карта сайта Kansoftware
НОВОСТИУСЛУГИРЕШЕНИЯКОНТАКТЫ
Разработка программного обеспечения

Iterator Adaptor

Boost , ,

Boost C++ Libraries

...one of the most highly regarded and expertly designed C++ library projects in the world. Herb Sutter and Andrei Alexandrescu, C++ Coding Standards

Iterator Adaptor

Author: Дэвид Абрахамс, Джереми Сиек, Томас Витт
Contact: iterator_adaptor шаблон класса адаптирует некоторые База1 тип для создания нового итератора. Мгновения iterator_adaptor происходят из соответствующего мгновенного сообщения iterator_facade и реализуют основное поведение в терминах типа Base. По сути, iterator_adaptor просто передает все операции на экземпляр типа Base, который он сохраняет в качестве члена.

[1](1, 2) Термин "База " здесь не относится к базовому классу и не означает использование деривации. Мы следовали примеру стандартной библиотеки, которая предоставляет функцию базового() для доступа к базовому итераторному объекту reverse_iterator адаптера.

Пользователь iterator_adaptor создает класс, полученный от мгновенной передачи iterator_adaptor, а затем выборочно переопределяет некоторые из основных функций членов, описанных в таблице iterator_facade. Тип База не должен отвечать полным требованиям к итератору; он должен поддерживать только операции, используемые основными функциями интерфейса iterator_adaptor, которые не были переопределены в производном классе пользователя.

Несколько параметров шаблона iterator_adaptor по умолчанию use_default. Это позволяет пользователю использовать параметр по умолчанию, даже если он хочет указать параметр позже в списке параметров. Кроме того, по умолчанию для соответствующих сопутствующих типов несколько сложны, поэтому метапрограммирование требуется для их вычисления, а use_default может помочь упростить реализацию. Наконец, личность типа use_default не остается неопределенной, поскольку спецификация помогает подчеркнуть, что параметр Reference шаблон может не всегда быть идентичным параметру итератора reference, и будет удерживать пользователей от ошибок на основе этого предположения.

Reference

template <
    class Derived
  , class Base
  , class Value               = use_default
  , class CategoryOrTraversal = use_default
  , class Reference           = use_default
  , class Difference = use_default
>
class iterator_adaptor
  : public iterator_facade<Derived, V', C', R', D'> // see details
{
    friend class iterator_core_access;
 public:
    iterator_adaptor();
    explicit iterator_adaptor(Base const& iter);
    typedef Base base_type;
    Base const& base() const;
 protected:
    typedef iterator_adaptor iterator_adaptor_;
    Base const& base_reference() const;
    Base& base_reference();
 private: // Core iterator interface for iterator_facade.
    typename iterator_adaptor::reference dereference() const;
    template <
    class OtherDerived, class OtherIterator, class V, class C, class R, class D
    >
    bool equal(iterator_adaptor<OtherDerived, OtherIterator, V, C, R, D> const& x) const;
    void advance(typename iterator_adaptor::difference_type n);
    void increment();
    void decrement();
    template <
        class OtherDerived, class OtherIterator, class V, class C, class R, class D
    >
    typename iterator_adaptor::difference_type distance_to(
        iterator_adaptor<OtherDerived, OtherIterator, V, C, R, D> const& y) const;
 private:
    Base m_iterator; // exposition only
};

iterator_adaptor requirements

static_cast (iterator_adaptor*) должны быть хорошо сформированы. Аргумент База должен быть знаковым и понятным.

iterator_adaptor base class parameters

V', C', R', и D' параметры iterator_facade, используемые в качестве базового класса в резюме iterator_adaptor выше, определены следующим образом:

V' = if (Value is use_default)
          return iterator_traits<Base>::value_type
      else
          return Value
C' = if (CategoryOrTraversal is use_default)
          return iterator_traversal<Base>::type
      else
          return CategoryOrTraversal
R' = if (Reference is use_default)
          if (Value is use_default)
              return iterator_traits<Base>::reference
          else
              return Value&
      else
          return Reference
D' = if (Difference is use_default)
          return iterator_traits<Base>::difference_type
      else
          return Difference

iterator_adaptor public operations

iterator_adaptor();

Requires:The Base type must be Default Constructible.
Returns:An instance of iterator_adaptor with m_iterator default constructed.

explicit iterator_adaptor(Base const& iter);

Returns:An instance of iterator_adaptor with m_iterator copy constructed from iter.

Base const& base() const;

Returns:m_iterator

iterator_adaptor protected member functions

База const& base_reference() const;

Returns:A const reference to m_iterator.

Base& base_reference();

Returns:A non-const reference to m_iterator.

iterator_adaptor private member functions

имя типа iterator_adaptor::reference dereference() const;

Returns:*m_iterator
template <
class OtherDerived, class OtherIterator, class V, class C, class R, class D
>
bool equal(iterator_adaptor<OtherDerived, OtherIterator, V, C, R, D> const& x) const;
Returns:m_iterator == x.base()

void advance(typename iterator_adaptor::difference_type n);

Effects:m_iterator += n;

void increment();

Effects:++m_iterator;

void decrement();

Effects:--m_iterator;
template <
    class OtherDerived, class OtherIterator, class V, class C, class R, class D
>
typename iterator_adaptor::difference_type distance_to(
    iterator_adaptor<OtherDerived, OtherIterator, V, C, R, D> const& y) const;
Returns:y.base() - m_iterator

Tutorial Example

В этом разделе мы продолжим работу над node_iter шаблон класса, который мы разработали в iterator_facade учебник. Если вы еще не читали этот материал, вы должны вернуться и проверить его, потому что мы собираемся забрать его прямо там, где он ушел.

Вы, вероятно, не подумали об этом так, но объект node_base*, который лежит в основе node_iterator, сам по себе является итератором, как и все другие указатели. Если внимательно изучить этот указатель с точки зрения итератора, то мы увидим, что он имеет много общего с node_iterator мы строим. Во-первых, они имеют большинство одинаковых типов ( value_type, reference, pointer, и difference_type). Во-вторых, даже некоторые основные функциональные возможности одинаковы: operator* и operator== на node_iterator возвратить результат ссылки на те же операции на основной указателя, через node_iterator dereference и equal функции участника. Единственная реальная поведенческая разница между node_base* и node_iterator может наблюдаться при их приращении: node_iterator следует за m_next указателя, в то время как node_base* просто применяет адресный офсет.

Оказывается, модель построения итератора на другом итераторном типе (База1 тип) при изменении всего нескольких аспектов поведения основного типа является чрезвычайно распространенным, и это шаблон, адресованный iterator_adaptor. Использование iterator_adaptor очень похоже на использование iterator_facade, но поскольку iterator_adaptor пытается имитировать как можно большую часть поведения Base типа, мы не должны поставлять аргумент Value, так и не реализовывать ни одно основное поведение, кроме инкремент>. Таким образом, реализация node_iter сводится к:

template <class Value>
class node_iter
  : public boost::iterator_adaptor<
        node_iter<Value>                // Derived
      , Value*                          // Base
      , boost::use_default              // Value
      , boost::forward_traversal_tag    // CategoryOrTraversal
    >
{
 private:
    struct enabler {};  // a private type avoids misuse
 public:
    node_iter()
      : node_iter::iterator_adaptor_(0) {}
    explicit node_iter(Value* p)
      : node_iter::iterator_adaptor_(p) {}
    template <class OtherValue>
    node_iter(
        node_iter<OtherValue> const& other
      , typename boost::enable_if<
            boost::is_convertible<OtherValue*,Value*>
          , enabler
        >::type = enabler()
    )
      : node_iter::iterator_adaptor_(other.base()) {}
 private:
    friend class boost::iterator_core_access;
    void increment() { this->base_reference() = this->base()->next(); }
};

Обратите внимание на использование node_iter::iterator_adaptor_ здесь: потому что iterator_adaptor определяет вложенный iterator_adaptor_ тип, который относится к себе, что дает нам удобный способ обратиться к сложному типу базового класса node_iter. [Примечание: эта техника, как известно, не работает с Borland C++ 5.6.4 и Metrowerks CodeWarrior до 9.0]

Вы можете увидеть примерную программу, которая использует эту версию узловых итераторов здесь.

В случае node_iter, не очень убедительно пройти boost::use_default как iterator_adaptor Value аргумент; мы могли бы просто пройти node_iter's Value вдоль it_adaptor, и даже короче! Большинство шаблонов класса итераторов, построенных с iterator_adaptor, имеют параметризацию на другом типе итератора, а не на его value_type. Например, boost::reverse_iterator принимает аргумент типа итератора и меняет направление движения, так как оригинальный итератор и обратное имеют все те же самые ассоциированные типы, iterator_adaptor делегирование типов по умолчанию Base сохраняет реализатора boost::reverse_iterator от написания:

std::iterator_traits<Iterator>::some-associated-type

по крайней мере четыре раза.

Мы настоятельно призываем вас пересмотреть документацию и реализацию reverse_iterator и других Boost специализированных адаптеров для генератора, чтобы получить представление о том, какие вещи вы можете сделать с iterator_adaptor. В частности, взгляните на transform_iterator, который является, пожалуй, самым простым адаптером, а также counting_iterator, что демонстрирует, что iterator_adaptor Base тип не должен быть итератором.

Статья Iterator Adaptor раздела может быть полезна для разработчиков на c++ и boost.




Материалы статей собраны из открытых источников, владелец сайта не претендует на авторство. Там где авторство установить не удалось, материал подаётся без имени автора. В случае если Вы считаете, что Ваши права нарушены, пожалуйста, свяжитесь с владельцем сайта.



:: Главная :: ::


реклама


©KANSoftWare (разработка программного обеспечения, создание программ, создание интерактивных сайтов), 2007
Top.Mail.Ru

Время компиляции файла: 2024-08-30 11:47:00
2025-05-20 00:08:29/0.027359008789062/1