How to implement standard iterators in class

前端 未结 3 1568
温柔的废话
温柔的废话 2021-02-06 04:40

I have classes which are usually using standard containers as underlying fields. For example, I have a class

template 
class Vec_3D
{
public:
          


        
相关标签:
3条回答
  • 2021-02-06 05:25

    std::iterator is a base class only, its basically a container for some traits, but if you wanted to use it to implement your own iterator class you'd need to derive from it.

    However you don't need to use it, there was a proposal to deprecate it, you could just define those traits directly in an iterator that you write. The following question has info on the proposal and help with implementing an iterator class:- Preparation for std::iterator Being Deprecated

    At the moment you're defining your container's iterator types using that base, not a class that can actually do any iterating, which is why it fails.

    You expose the array as a public member. If you're happy to expose that your vec_3d is implemented using an array (whether you continue to expose the member array publicly or not) then you could just use the array's iterators - its not clear from the question that your iterator needs any bespoke behaviour just because your container adds some functionality.

    0 讨论(0)
  • 2021-02-06 05:35

    std::iterator is (was) a helper type to define the typedefs that a typical iterator requires. These typedefs within the class in turn make std::iterator_traits work with your iterator.

    It does not, however, actually implement the required operations for you.

    It was deprecated, because the std committee didn't like specifying that standard iterators had to have those typedefs, and writing the typedefs was not much bulkier than figuring out what arguments to pass to the std::iterator template.

    The easy thing to do here is to just steal your underlying container's iterator. This makes your abstraction leak, but it is efficient and easy.

    template <typename T>
    struct Vec_3D {
      using container=std::array<T, 3>;
      using iterator=typename container::iterator;
      using const_iterator=typename container::const_iterator;
    
      iterator begin() { return vec.begin(); }
      iterator end() { return vec.end(); }
      const_iterator begin() const { return vec.begin(); }
      const_iterator end() const { return vec.end(); }
    private:
      /* ... */
      container vec;
      /* ... */
    };
    

    If you don't want to expose your underlying container type, if you are willing to guarantee your underlying container is a contiguous buffer you can do:

    template <typename T>
    struct Vec_3D {
      using iterator=T*;
      using const_iterator=T const*;
    
      iterator begin() { return vec.data(); }
      iterator end() { return vec.data()+vec.size(); }
      const_iterator begin() const { return vec.data(); }
      const_iterator end() const { return vec.data()+vec.size(); }
    private:
      /* ... */
      std::array<T,3> vec;
      /* ... */
    };
    

    as pointers are valid iterators.

    If you find you are writing this "I am a modified container" boilerplate too much, you can automate it:

    template<class Container>
    struct container_wrapper {
      using container=Container;
    
      using iterator=typename container::iterator;
      using const_iterator=typename container::const_iterator;
    
      iterator begin() { return m_data.begin(); }
      iterator end() { return m_data.end(); }
      const_iterator begin() const { return m_data.begin(); }
      const_iterator end() const { return m_data.end(); }
    protected:
      Container m_data;
    };
    

    and then

    template <typename T>
    class Vec_3D:private container_wrapper<std::array<T,3>> {
      // ...
    };
    

    but even that might be a bit much, why not just:

    template <typename T>
    class Vec_3D:public std::array<T,3> {
      // ...
    };
    

    It is true that deleting Vec_3D through a pointer to base is undefined behavior, but who deletes pointers to standard containers?

    If this worries you:

    template <typename T>
    class Vec_3D: private std::array<T,3> {
      using container = std::array<T,3>;
      using container::begin();
      using container::end();
      // ...
    };
    

    lets you inherit privately, then bring certain operations back into scope.

    0 讨论(0)
  • 2021-02-06 05:37

    A range-based for loop only requires that your class have begin() and end() methods (or overloads of std::begin() and std::end()) that return iterators. It doesn't care where those iterators come from. So, the simplest solution is to just use the array's own iterators instead of trying to define your own, eg:

    template <typename T>
    class Vec_3D
    {
    public:
        typedef typename std::array<T, 3> array_type;
        typedef typename array_type::iterator iterator;
        typedef typename array_type::const_iterator const_iterator;
        // or:
        // using array_type = std::array<T, 3>;
        // using iterator = array_type::iterator;
        // using const_iterator = array_type::const_iterator;
        ...
    
        inline iterator begin() noexcept { return vec.begin(); }
        inline const_iterator cbegin() const noexcept { return vec.cbegin(); }
        inline iterator end() noexcept { return vec.end(); }
        inline const_iterator cend() const noexcept { return vec.cend(); }
        ...
    
    private:
        array_type vec;
    };
    
    0 讨论(0)
提交回复
热议问题