不错的问题。我会尝试实现一个特定的迭代器包装类型,将两个范围变成一个范围。东西的线路:
// compacted syntax for brevity...
template <typename T1, typename T2>
struct concat_iterator
{
public:
typedef std::forward_iterator_tag iterator_category;
typedef typename iterator_traits<T1>::value_type value_type;
typedef *value_type pointer;
typedef &value_type reference;
concat_iterator(T1 b1, T1 e1, T2 b2, T2 e2)
: seq1(b1), seq1end(e1), seq2(b2), seq2end(e2);
iterator& operator++() {
if (seq1 != seq1end) ++seq1;
else ++seq2;
return this;
}
reference operator*() {
if (seq1 != seq1end) return *seq1;
else return *seq2;
}
pointer operator->() {
if (seq1 != seq1end) return &(*seq1);
else return &(*seq2);
}
bool operator==(concat_iterator const & rhs) {
return seq1==rhs.seq1 && seq1end==rhs.seq2
&& seq2==rhs.seq2 && seq2end==rhs.seq2end;
}
bool operator!=(contact_iterator const & rhs) {
return !(*this == rhs);
}
private:
T1 seq1;
T1 seq1end;
T2 seq2;
T2 seq2end;
};
template <typename T1, typename T2>
concat_iterator<T1,T2> concat_begin(T1 b1, T1 e1, T2 b2, T2 e2)
{
return concat_iterator<T1,T2>(b1,e1,b2,e2);
}
template <typename T1, typename T2>
concat_iterator<T1,T2> concat_end(T1 b1, T1 e1, T2 b2, T2 e2)
{
return concat_iterator<T1,T2>(e1,e1,e2,e2);
}
现在你可以使用:
class X {
public:
template <typename Iter, typename Iter2>
X(Iter b1, Iter e1, Iter2 b2, Iter2 e2)
: mVec(concat_begin(b1,e1,b2,e2), concat_end(b1,e1,b2,e2))
{}
private:
vector<Y> const mVec;
};
或(我刚刚想到这一点)你不需要重新声明构造函数。让你的来电者使用帮手功能:
X x(concat_begin(b1,e1,b2,e2), concat_end(b1,e1,b2,e2));
我还没有检查过代码,只是在这里输入了我的头顶。它可以编译或不能,它可以工作或不可以...但你可以把它作为一个起点。
在我的情况下,构造实例后,vector成员变量不应该改变。使它成为const可以帮助编译器帮助我保证。 – SCFrench 2009-04-16 17:39:37
好吧,考虑到进行连接所需的代码量,如果你保持const,那么更有可能你的代码会有bug。 – avakar 2009-04-16 17:53:53
SCFrench,是不是已经足够安全了X :: mvec在X被构造后不会改变? – veefu 2009-04-16 18:51:02