迭代封装的嵌套STL容器

Sim*_*ott 6 c++ stl

这是一个相当普通的STL容器封装,它允许Cfoo的用户迭代容器而不允许更改内部.

#include <vector>

class Cfoo
{
public:
    class Cbar
    {
        /* contents of Cbar */
    };
    typedef std::vector<Cbar> TbarVector;
    typedef TbarVector::const_iterator const_iterator;     
public:
    const_iterator begin() const { return( barVector_.begin() ); }
    const_iterator end() const { return( barVector_.end() ); }
private:
    TbarVector barVector_;
};
Run Code Online (Sandbox Code Playgroud)

到现在为止还挺好.我们可以像这样迭代容器:

Cfoo myFoo;
for (Cfoo::const_iterator it = myFoo.begin(); it != myFoo.end(); ++it)
{
   it->DoSomething();
}
Run Code Online (Sandbox Code Playgroud)

现在我想用一个嵌套的std :: vector替换std :: vector:

public:
    typedef std::vector<Cbar> TbarVectorInner;
    typedef std::vector<TbarVectorInner> TbarVectorOuter;

private:
    TbarVectorOuter barContainer_;
Run Code Online (Sandbox Code Playgroud)

但我希望能够以与以前相同的方式迭代Cbar的所有实例,公开const_iterator,以及begin()const和end()const方法.

我不清楚如何做到这一点,虽然我怀疑它涉及编写自定义迭代器.有什么想法吗?

Mar*_*som 4

没有一个标准迭代器能够迭代多个容器,因此您的假设是正确的 - 您必须编写一个自定义迭代器。

如果您有一个将(开始,结束)迭代器对返回到内部容器的中间迭代器,则可以以通用方式执行此操作。

一些未经测试的入门代码:

template<typename T, typename OuterIterator, typename InnerIterator>
class Iterator2d : std::Iterator
{
public:
    Iterator2d(OuterIterator begin, OuterIterator end) : m_begin(begin), m_end(end), m_currentOuter(begin)    {
        if (m_currentOuter != m_end)
             m_currentInner = m_begin->first;
        Normalize();
    }
    Iterator2d & operator++()
    {
        if (m_currentOuter != m_end)
        {
            ++m_currentInner;
            Normalize();
        }
        return *this;
    }
    T & operator*()
    {
        return *m_currentInner;
    }
private:
    void Normalize()
    {
        while (m_currentOuter != m_end && m_currentInner == m_currentOuter->second)
        {
            ++m_currentOuter;
            if (m_currentOuter != m_end)
                m_currentInner = m_currentOuter->first;
        }
    }

    OuterIterator m_begin;
    OuterIterator m_end;
    OuterIterator m_currentOuter;
    InnerIterator m_currentInner;
};
Run Code Online (Sandbox Code Playgroud)

这只是一个开始,我可能会回来完成它 - 也可能不会,具体取决于此实现已经涵盖了相同的基础。