4

Tのような型がある場合、を取得するFoo<mpl::_1>ことができます。mpl::apply<T, int>::typeFoo<int>

しかし、T例えば のような完全な型の場合は、Foo<int>コンパイルmpl::apply<T, int>::typeされません。

可能な場合は型を適用し、そうでない場合は型を返すメタ関数を作成したいと考えています。次のようなものです:

template <typename Partial, typename T>
struct maybe_apply
: eval_if_c<??????,
    mpl::apply<Partial, T>,
    mpl::identity<Partial>>
{ };

これが私が望むことをするために、???sに何を入れることができますか?

4

1 に答える 1

2

免責事項: 私は MPL の専門家とはほど遠いので、これがこれを解決するための最良の方法であるとは約束できません (または、それが正しいとしても、少なくとも動作するようです)。

ドキュメントによると、最初のパラメーター/引数mpl::applyはラムダ式である必要があり、メタ関数クラスまたはプレースホルダー式のいずれかにすることができます。簡単なグーグル検索で、この投稿にたどり着きました。その投稿によるとmpl::is_lambda_expression、タイプがプレースホルダー式であるかどうかを判断できます。Boost.TTI (バージョン 1.54 以降のブースト) では、まさに必要なことを行うメタ関数を見つけることができます。このメタ関数はboost::tti::detail::is_lambda_expressionにありますboost/tti/detail/dlambda.hpp。以下の例では、TTI が型が Metafunction クラスであるかどうかを検出するために使用するマクロとまったく同じものを使用しています。

Coliruで実行しています。

#include <iostream>
#include <typeinfo>

#include <boost/utility.hpp>

#include <boost/mpl/apply.hpp>
#include <boost/mpl/eval_if.hpp>
#include <boost/mpl/identity.hpp>

#include <boost/mpl/plus.hpp>

namespace mpl=boost::mpl;


/* //This is another way to do it
template <typename T, typename Enable=void>
struct is_apply_able : mpl::false_
{};

template <typename T>
struct is_apply_able<T,typename boost::enable_if<mpl::is_lambda_expression<T> >::type> : mpl::true_
{};
BOOST_MPL_HAS_XXX_TEMPLATE_NAMED_DEF(is_metafunction_class, apply, false)
template <typename T>
struct is_apply_able<T,typename boost::enable_if<is_metafunction_class<T> >::type> : mpl::true_
{};*/


BOOST_MPL_HAS_XXX_TEMPLATE_NAMED_DEF(is_metafunction_class, apply, false)

template <typename T>
struct is_apply_able : mpl::or_<is_metafunction_class<T>,mpl::is_lambda_expression<T> >
{};

struct plus_two
{
    template <typename Number>
    struct apply
    {
        typedef typename mpl::plus<Number,mpl::int_<2> >::type type;
    };
};

template <typename T>
struct Foo
{};

template <typename Partial, typename T>
struct maybe_apply
: mpl::eval_if<is_apply_able<Partial>,
    mpl::apply<Partial, T>,
    mpl::identity<Partial> > 
{ };

int main()
{
    std::cout << typeid(maybe_apply<Foo<mpl::_1>,int>::type).name() << std::endl;
    std::cout << typeid(maybe_apply<plus_two,mpl::int_<1> >::type).name() << std::endl;
    std::cout << typeid(maybe_apply<Foo<float>,int>::type).name() << std::endl;
}
于 2013-09-01T20:15:58.440 に答える