如何在运行时在boost::fusion::vector中查找元素

How to find an element in a boost::fusion::vector at runtime?

本文关键字:查找 元素 vector 运行时 boost fusion      更新时间:2023-10-16

我这里有一个通用状态机的专有实现,它使用std::tr1::tuple作为转换表:

template<State StartState, Event TriggerEvent, State TargetState>
struct transition {...};
typedef std::tr1::tuple< transition< ready      , run      , running     >
                       , transition< running    , terminate, terminating >
                       , transition< terminating, finish   , terminated  >
                       > transition_table;

有一个功能

template<typename Transitions>
State find_next_state( State current
                     , Event event
                     , const Transitions& transition_table );

在给定当前状态和事件的情况下,在转换表中查找下一个状态。

除了这个平台的tuple实现不支持超过10个项目之外,这一切都很好。boost::tuple似乎也是如此,所以我尝试使用boost::fusion::vector。但fusion的find_if似乎只采用"一元MPL-Lambda表达式"——我想,它只在编译时起作用。

既然如此,我该如何实现find_next_state()呢?

注意:

这是一个专有的嵌入式平台,只提供GCC 4.1.2,所以我们只能使用C++03+TR1。

编写自己的find_if非常简单,除了"返回找到的值"部分。由于boost::fusion::vector是一个异构容器,因此没有一个正确的类型可以返回。脑海中出现的一个可能的解决方案是接受一个用找到的值调用的延续函数:

#include <boost/fusion/include/size.hpp>
#include <boost/fusion/include/at_c.hpp>
// private implementation details
namespace detail{
// shorthand ...
template<class S>
struct fusion_size{
  static const unsigned value =
    boost::fusion::result_of::size<S>::type::value;
};
// classic compile-time counter
template<unsigned> struct uint_{};
template<class Seq, class Pred, class F>
void find_if(Seq&, Pred const&, F, uint_<fusion_size<Seq>::value>, int)
{ /* reached the end, do nothing */ }
template<class Seq, class Pred, class F, unsigned I>
void find_if(Seq& s, Pred const& pred, F f, uint_<I>, long){
    if(pred(boost::fusion::at_c<I>(s)))
    {
        f(boost::fusion::at_c<I>(s));
        return; // bail as soon as we find it
    }
    find_if(s, pred, f, uint_<I+1>(), 0);
}
} // detail::
template<class Seq, class Pred, class F>
void find_if(Seq& s, Pred const& pred, F f){
    detail::find_if(s, pred, f, detail::uint_<0>(), 0);
}

活生生的例子。

I+1==fusion_size<Seq>::value时,intlong参数以及0参数只是为了消除歧义,因为这两个函数都同样可行。CCD_ 14是类型CCD_。