Lar*_*ars 6 c++ tuples boost-mpl c++11
该boost::mpl
算法似乎不能够在工作std::tuple
类型的开箱即用的例如,下列不编译(升压1.46.0,G ++快照2011-02-19):
#include <tuple>
#include <boost/mpl/vector.hpp>
#include <boost/mpl/contains.hpp>
namespace mpl=boost::mpl;
typedef mpl::vector<int,float,bool> types;
static_assert(mpl::contains<types, float>::value, "vector contains bool");
typedef std::tuple<int,float,bool> types2;
// the following does not compile:
// error: no class template named ‘apply’ in ‘struct boost::mpl::contains_impl<boost::mpl::non_sequence_tag>’
static_assert(mpl::contains<types2, float>::value, "tuple contains bool");
Run Code Online (Sandbox Code Playgroud)
使boost::mpl
算法工作的最简单方法是什么std::tuple
?
boost::fusion
提供此功能(因为它这样做boost::tuple
)?boost::tuple
以std::tuple
容易吗?小智 7
如果你不想将std :: tuple转换为mpl类型,你可以重载标签调度boost mpl使用:
#include <tuple>
#include <boost/mpl/sequence_tag.hpp>
#include <boost/mpl/pop_front_fwd.hpp>
#include <boost/mpl/push_front_fwd.hpp>
#include <boost/mpl/push_back_fwd.hpp>
#include <boost/mpl/front_fwd.hpp>
#include <boost/mpl/empty_fwd.hpp>
#include <boost/mpl/size_fwd.hpp>
#include <boost/mpl/at_fwd.hpp>
#include <boost/mpl/back_fwd.hpp>
#include <boost/mpl/clear_fwd.hpp>
#include <boost/mpl/pop_back_fwd.hpp>
#include <boost/mpl/iterator_tags.hpp>
#include <boost/mpl/next_prior.hpp>
#include <boost/mpl/deref.hpp>
#include <boost/mpl/begin_end_fwd.hpp>
namespace boost { namespace mpl {
namespace aux { struct std_tuple; }
template<class ... Args>
struct sequence_tag<std::tuple<Args...> >
{
typedef aux::std_tuple type;
};
template<>
struct front_impl< aux::std_tuple >
{
template< typename Tuple > struct apply
: std::tuple_element<0, Tuple>
{
};
};
template<>
struct empty_impl< aux::std_tuple >
{
template< typename Tuple > struct apply
: std::integral_constant<bool, std::tuple_size<Tuple>::value == 0>
{
};
};
template<>
struct pop_front_impl< aux::std_tuple >
{
template< typename Tuple > struct apply;
template< class First, class ... Types > struct apply<std::tuple<First, Types...>>
{
typedef std::tuple<Types...> type;
};
};
template<>
struct push_front_impl< aux::std_tuple >
{
template< typename Tuple, typename T > struct apply;
template< typename T, typename ... Args >
struct apply<std::tuple<Args...>, T>
{
typedef std::tuple<T, Args...> type;
};
};
template<>
struct push_back_impl< aux::std_tuple >
{
template< typename Tuple, typename T > struct apply;
template< typename T, typename ... Args >
struct apply<std::tuple<Args...>, T>
{
typedef std::tuple<Args..., T> type;
};
};
template<>
struct size_impl< aux::std_tuple >
{
template< typename Tuple > struct apply
: std::tuple_size<Tuple>
{
};
};
template<>
struct at_impl< aux::std_tuple >
{
template< typename Tuple, typename N > struct apply
: std::tuple_element<N::value, Tuple>
{
};
};
template<>
struct back_impl< aux::std_tuple >
{
template< typename Tuple > struct apply
: std::tuple_element<std::tuple_size<Tuple>::value - 1, Tuple>
{
};
};
template<>
struct clear_impl< aux::std_tuple >
{
template< typename Tuple > struct apply
{
typedef std::tuple<> type;
};
};
template<>
struct pop_back_impl< aux::std_tuple >
{
template<int ...> struct tuple_seq {};
template<int N, int ...S> struct tuple_gens : tuple_gens<N-1, N-1, S...> {};
template<int ...S> struct tuple_gens<0, S...>{ typedef tuple_seq<S...> type; };
template < class Tuple, class Index> struct apply_impl;
template < class Tuple, int ... S> struct apply_impl<Tuple, tuple_seq<S...>>
{
typedef std::tuple<typename std::tuple_element<S, Tuple>::type...> type;
};
template< typename Tuple > struct apply : apply_impl<Tuple, typename tuple_gens<std::tuple_size<Tuple>::value - 1>::type> { };
};
template< class ... Args >
struct tuple_iter;
template< class ... Args >
struct tuple_iter<std::tuple<Args...>>
{
typedef aux::std_tuple tag;
typedef forward_iterator_tag category;
};
template<>
struct begin_impl< aux::std_tuple >
{
template< class Tuple > struct apply
{
typedef tuple_iter<Tuple> type;
};
};
template<>
struct end_impl< aux::std_tuple >
{
template< typename > struct apply
{
typedef tuple_iter<std::tuple<>> type;
};
};
template< typename First, class ... Args >
struct deref< tuple_iter<std::tuple<First, Args...> > >
{
typedef First type;
};
template< typename First, class ... Args >
struct next< tuple_iter<std::tuple<First, Args...>> >
{
typedef tuple_iter< std::tuple<Args...> > type;
};
} }
Run Code Online (Sandbox Code Playgroud)
以及相关的测试:
#include <boost/mpl/pop_front.hpp>
#include <boost/mpl/push_front.hpp>
#include <boost/mpl/push_back.hpp>
#include <boost/mpl/front.hpp>
#include <boost/mpl/empty.hpp>
#include <boost/mpl/size.hpp>
#include <boost/mpl/at.hpp>
#include <boost/mpl/back.hpp>
#include <boost/mpl/clear.hpp>
#include <boost/mpl/pop_back.hpp>
#include <boost/mpl/contains.hpp>
#include <boost/mpl/aux_/test.hpp>
MPL_TEST_CASE()
{
typedef std::tuple<int, char, bool> Tuple;
MPL_ASSERT((is_same<front<Tuple>::type, int>));
MPL_ASSERT_RELATION( size<Tuple>::type::value, ==, 3 );
MPL_ASSERT(( is_same< pop_front<Tuple>::type, std::tuple<char, bool> > ));
MPL_ASSERT(( is_same< push_front<Tuple, unsigned>::type, std::tuple<unsigned, int, char, bool> > ));
MPL_ASSERT(( is_same< push_back<Tuple, unsigned>::type, std::tuple<int, char, bool, unsigned> > ));
MPL_ASSERT_RELATION( empty<Tuple>::type::value, ==, false );
MPL_ASSERT(( is_same< at_c<Tuple, 0>::type, int > ));
MPL_ASSERT(( is_same< at_c<Tuple, 1>::type, char > ));
MPL_ASSERT(( is_same< back<Tuple>::type, bool > ));
MPL_ASSERT(( is_same< clear<Tuple>::type, std::tuple<> > ));
MPL_ASSERT(( is_same< pop_back<Tuple>::type, std::tuple<int, char> > ));
MPL_ASSERT(( contains<Tuple, int> ));
MPL_ASSERT(( contains<Tuple, char> ));
MPL_ASSERT(( contains<Tuple, bool> ));
MPL_ASSERT_NOT(( contains<Tuple, unsigned> ));
}
Run Code Online (Sandbox Code Playgroud)
我用gcc 4.7.2和clang 3.2测试了这个.它应该包含在mpl中使用任何东西所需的一切(实际上它比它需要的更多).您可以将元组视为mpl :: list(转发迭代器编译类型).所以,为了让它发挥得更好你应该实现boost :: mpl :: list的功能.快速浏览boost/mpl/list/aux_目录:begin_end.hpp empty.hpp iterator.hpp pop_front.hpp push_back.hpp size.hpp clear.hpp front.hpp push_front.hpp tag.hpp是需要的相关文件将要执行.
从 std::tuple 转换为 boost 类型并返回似乎是最简单的方法
#include <iostream>
#include <tuple>
#include <type_traits>
#include <boost/mpl/if.hpp>
#include <boost/mpl/not.hpp>
#include <boost/mpl/vector.hpp>
namespace mpl = boost::mpl;
template<typename Sequence, typename T>
struct push_front;
template<template<typename...> class Sequence, typename T, typename ... Args>
struct push_front< Sequence<Args...>,T> {
typedef Sequence<T, Args...> type;
};
template<template<typename...> class To, typename From> struct tuple_change;
template<template<typename...> class To, template<typename...> class From, typename ... Args>
struct tuple_change<To, From<Args...>>
{
typedef To<Args...> type;
};
template<typename Sequence, size_t N>
struct at : std::tuple_element<N,Sequence> { };
template<typename Sequence>
struct empty;
template<template<typename...> class Sequence, typename ... Args>
struct empty<Sequence<Args...>> {
typedef Sequence<> type;
};
template<
size_t N,
typename Sequence,
template<typename> class Pred,
typename ... Args >
struct while_impl
{
typedef typename mpl::if_c<
Pred<
typename at<Sequence, sizeof...(Args) - N -1>::type
>::value,
typename push_front<
typename while_impl<N-1, Sequence, Pred, Args...>::type,
typename at<Sequence,sizeof...(Args)-N-1>::type
>::type,
typename empty< Sequence > ::type
>::type type;
};
template<
typename Sequence,
template<typename> class Pred,
typename ... Args >
struct while_impl<-1, Sequence, Pred, Args...>
: empty<Sequence> {
};
template<
typename Sequence,
template<typename> class Pred>
struct while_;
template<
template<typename...> class Sequence,
template<typename> class Pred,
typename ... Args >
struct while_< Sequence<Args...>, Pred >
{
typedef typename while_impl<sizeof...(Args)-1, Sequence<Args...>, Pred, Args...>::type type;
};
template<typename T>
struct not_na : mpl::not_< std::is_same<mpl_::na, T> >
{ };
template<template<typename...> class To, typename From>
struct to_boost;
template<template<typename...> class To, typename...Args >
struct to_boost<To, std::tuple<Args...> > :
tuple_change< mpl::vector, std::tuple<Args...> >
{ };
template< typename From >
struct to_std;
template<template<typename...> class From, typename...Args >
struct to_std< From<Args...> > :
while_<typename tuple_change< std::tuple, From<Args...> >::type, not_na>
{ };
static_assert(
std::is_same<
mpl::vector< char, int, bool>,
typename to_boost<mpl::vector, std::tuple<char, int, bool> >::type
>::value,
"tuple_change to boost failed");
static_assert(
std::is_same<
std::tuple< char, int, bool>,
typename to_std< mpl::vector<char, int, bool> >::type
>::value,
"tuple_change from boost failed");
int main(){ return 0;}
Run Code Online (Sandbox Code Playgroud)
*测试使用:
MacOSx 上的 boost_1_46_0 和 g++-4.5
boost_1_45_0 和 Ubuntu 10.10 上的 g++-4.5