【问题标题】:Custom iterator does not work with BOOST_FOREACH?自定义迭代器不适用于 BOOST_FOREACH?
【发布时间】:2011-05-03 00:13:26
【问题描述】:

我有一个包含一些数据的类,我想添加 begin()end() 函数,它们提供数据 ID 的迭代器。

我正在使用 Boost counting_iterator:

#include <iostream>
#include <vector>
#include <boost/foreach.hpp>
#include <boost/iterator/counting_iterator.hpp>

template<class T>   
class ContainerTpl {
 public:
  typedef std::size_t Id;
  typedef boost::counting_iterator<Id> const_iterator;
  ContainerTpl() {}
  const_iterator begin() {
    return boost::counting_iterator<Id>(0);
  }
  const_iterator end() {
    return boost::counting_iterator<Id>(container_.size());
  }
 private:
  std::vector<T> container_;
};

int main () {
  typedef ContainerTpl<double> Container;
  Container c;
  BOOST_FOREACH (Container::Id cid, c) {
    std::cerr << cid << std::endl;
  }
  return 0;
}

请注意,这是最小的示例代码;实际上,该类包含更多功能,因此,例如,typedefvector 是不够的。我真的需要那个类,它带有一个对 ID 的迭代器。

不幸的是,上面的代码给了我非常讨厌的编译器错误:

In file included from boost/foreach.hpp:71,
                 from a.cpp:3:
boost/mpl/eval_if.hpp: In instantiation of ‘boost::mpl::eval_if<mpl_::bool_<false>, boost::range_const_iterator<ContainerTpl<double> >, boost::range_mutable_iterator<ContainerTpl<double> > >’:
boost/foreach.hpp:355:   instantiated from ‘boost::foreach_detail_::foreach_iterator<ContainerTpl<double>, mpl_::bool_<false> >’
a.cpp:25:   instantiated from here
boost/mpl/eval_if.hpp:38: error: no type named ‘type’ in ‘struct boost::range_mutable_iterator<ContainerTpl<double> >’
a.cpp: In function ‘int main()’:
a.cpp:25: error: no matching function for call to ‘begin(const boost::foreach_detail_::auto_any_base&, boost::foreach_detail_::type2type<ContainerTpl<double>, mpl_::bool_<false> >*, boost::mpl::o\
r_<boost::mpl::and_<boost::mpl::not_<boost::is_array<ContainerTpl<double> > >, mpl_::bool_<false>, mpl_::bool_<true>, mpl_::bool_<true>, mpl_::bool_<true> >, boost::mpl::and_<boost::mpl::not_<boo\
st::foreach::is_noncopyable<ContainerTpl<double> > >, boost::foreach::is_lightweight_proxy<ContainerTpl<double> >, mpl_::bool_<true>, mpl_::bool_<true>, mpl_::bool_<true> >, mpl_::bool_<false>, m\
pl_::bool_<false>, mpl_::bool_<false> >*)’
a.cpp:25: error: no matching function for call to ‘end(const boost::foreach_detail_::auto_any_base&, boost::foreach_detail_::type2type<ContainerTpl<double>, mpl_::bool_<false> >*, boost::mpl::or_\
<boost::mpl::and_<boost::mpl::not_<boost::is_array<ContainerTpl<double> > >, mpl_::bool_<false>, mpl_::bool_<true>, mpl_::bool_<true>, mpl_::bool_<true> >, boost::mpl::and_<boost::mpl::not_<boost\
::foreach::is_noncopyable<ContainerTpl<double> > >, boost::foreach::is_lightweight_proxy<ContainerTpl<double> >, mpl_::bool_<true>, mpl_::bool_<true>, mpl_::bool_<true> >, mpl_::bool_<false>, mpl\
_::bool_<false>, mpl_::bool_<false> >*)’
a.cpp:25: error: no matching function for call to ‘deref(const boost::foreach_detail_::auto_any_base&, boost::foreach_detail_::type2type<ContainerTpl<double>, mpl_::bool_<false> >*)’

我怎样才能使代码工作?

更新:在回答之后,我添加了以下代码,使其工作:

namespace boost
{
// specialize range_mutable_iterator and range_const_iterator in                                                                                                                                    
// namespace boost                                                                                                                                                                                  
template<class T>
struct range_mutable_iterator< ContainerTpl<T> > {
  typedef typename ContainerTpl<T>::const_iterator type;
};

template<class T>
struct range_const_iterator< ContainerTpl<T> > {
  typedef typename ContainerTpl<T>::const_iterator type;
};
} // end namespace         

【问题讨论】:

    标签: boost iterator


    【解决方案1】:

    Boost 文档中有一个关于此的页面:

    http://boost-sandbox.sourceforge.net/libs/foreach/doc/html/foreach/extensibility.html

    简而言之,您需要为您的类型定义boost::range_mutable_iterator&lt;&gt;,以便编译器可以实例化BOOST_FOREACH 尝试使用的模板类型。

    为未来的谷歌员工编辑:

    我不确定该“沙盒”网址是否始终指向最新版本,或者是最终会中断的临时位置。这是当前版本的链接,可能会更稳定,但会过时:

    http://www.boost.org/doc/libs/1_50_0/doc/html/foreach/extensibility.html

    【讨论】:

      【解决方案2】:

      我知道这是一个老问题,但似乎仍然相关。我遇到了这个问题,并注意到 boost foreach 要求你有一个可变的迭代器,以及一个定义的 const_iterator(因此,ContainerTpl::iterator 和 ContainerTpl::const_iterator)。否则,您需要按照 Tim 提供的说明进行操作。

      【讨论】:

      • 这对于它的工作是必不可少的。在我的情况下,typedef iterator const_iterator 就足够了,但它需要在那里。否则你会得到神秘的编译器错误:/
      猜你喜欢
      • 1970-01-01
      • 2019-01-04
      • 2019-10-31
      • 1970-01-01
      • 2010-12-15
      • 2012-11-17
      • 1970-01-01
      • 2015-02-04
      相关资源
      最近更新 更多