基于部分模板模板的专用化与显式部分模板专用化
Partial template-template based specialization vs explicit partial template specialization
给定一个模板,例如foo
:
template<typename... ARGS>
struct foo {};
模板的两个部分专用化bar
:
template<template<typename...> class T , typename... ARGS>
struct bar<T<ARGS...>>
{};
template<typename... ARGS>
struct bar<foo<ARGS...>>
{};
第二个部分专用化不是比第一个更专业,并且必须实例化而不是模板模板参数专用化吗?
一些背景:
我目前正在基于本文为模板元编程编写多变量 lambda 表达式。
正如论文所示,给定类似 Haskell 的 let
表达式,人们可以很容易地开发 tmp lambda 表达式。就我而言,我已经扩展了论文的内容,开发基于可变变量 let 表达式的可变模板(通过 curryfing 多个嵌套的一元 let 表达式(,然后实现多变量 lambda 表达式。
我的 lambda 表达式模板 tml::multi_lambda
定义如下:
template<typename BODY , typename... VARIABLES>
struct multi_lambda
{
template<typename... ARGS>
using result = tml::eval<tml::multi_let<VARIABLES...,
ARGS...,
BODY
>>;
};
其中tml::eval
是用于评估表达式的元函数,例如 Boost::mpl mpl::apply
(有关更多上下文,请参阅我之前的问题(。
评估函数tml::eval
既专用于泛型函数表达式,也专门用于此 lambda 表达式。这是上述示例的两个专业。
当我尝试计算 lambda 表达式时,例如:
using lambda = tml::multi_lambda<_1,_2, f<_1,_2>>; //f is a function,
//_1 _2 are placeholders
using result = tml::eval<lambda,int,int>; //Evaluate lambda with int int as parameters
tml::eval
实例化通用模板模板专用化(专为通用可评估表达式设计(,而不是 lambda 的部分专用化。
编辑:实施tml::eval
和SSCCE
tml::eval
是一个元函数,用于计算任何类型的表达式,返回结果。默认实现专用于三种情况:
- 表达式
不是一个函数,只是一个值:计算此类表达式的结果是表达式本身:
template<typename E> struct evaluate_impl<E> { using result = E; };
表达式是一个函数:计算的结果是函数的
result
成员类型的值。函数的参数也被计算(照顾嵌套表达式(:template<template<typename...> class F , typename... ARGS> struct evaluate_impl<F<ARGS...>> : public F<tml::eval<ARGS>...> {};
表达式是一个函数,并且将更多参数传递给
tml::eval
以使用该自定义参数计算表达式: 表达式的参数将被忽略,并传递和计算自定义:template<template<typename...> class F , typename... PLACEHOLDERS , typename... ARGS> struct evaluate_impl<F<PLACEHOLDERS...>,ARGS...> : public F<tml::eval<ARGS>...> {};
所以tml::eval
只是一个模板别名来骑在typename ::result
上:
template<typename... ARGS>
using eval = typename eval_impl<ARGS...>::result;
最后,用户可以专门化eval_impl
来覆盖该默认行为,或使极端情况起作用。例如,我的单变量 lambda 表达式模板,定义如下:
template<typename VARIABLE , typename VALUE , typename BODY>
struct lambda
{
template<typename ARG>
using result = tml::eval<tml::let<VARIABLE , ARG , BODY>>;
};
专门eval_impl
使 lambda 表达式的计算工作:
template<typename VARIABLE , typename BODY , typename ARG>
struct evaluate_impl<tml::lambda<VARIABLE,BODY>,ARG>
{
using result = typename tml::lambda<VARIABLE,BODY>::template result<ARG>;
};
我遇到的多变量 lambda 表达式采用类似的方法:
template<typename... VARIABLES , typename BODY , typename... ARG>
struct evaluate_impl<tml::multi_lambda<BODY,VARIABLES...>,ARGS...>
{
using result = typename tml::multi_lambda<BODY,VARIABLES...>::template result<ARGS...>;
};
但是,tml::eval
不是工作(如一个变量对应项(,而是实例化默认evaluate_impl
实现的第三种情况,或者由于不明确的专用化而失败(情况三与multi_lambda
专用化(。
这是一个SSCCE:
//An example function:
template<typename... ARGS>
struct F
{
using result = std::integral_constant<std::size_t,sizeof...(ARGS)>;
};
//This works fine:
using lambda_1 = tml::lambda<_1,F<_1,_1,_1,_1>>;
using result_1 = tml::eval<lambda_1,int>; //Call the lambda with int as parameter
//This doesn't work:
using lambda_2 = tml::multi_lambda<_1,_2,F<_1,_1,_2,_2>>;
using result_2 = tml::eval<lambda_2,int,int>; //Call the lambda with two int as parameters.
lambda_2
的评估失败,并显示:
functional.hpp:167:76: error: ambiguous class template instantiation for 'struct tml::impl::evaluate_impl<tml::impl::multi_lambda<tml::placeholders::_1, tml::placeholders::_2, f<tml::placeholders::_1, tml::placeholders::_1, tml::placeholders::_2, tml::placeholders::_2> >, int, int>'
using eval = typename impl::evaluate_impl<EXPRESSION , ARGS...>::result;
^
functional.hpp:116:16: error: candidates are: struct tml::impl::evaluate_impl<F<PLACEHOLDERS ...>, ARG, ARGS ...>
struct evaluate_impl<F<PLACEHOLDERS...> , ARG , ARGS...> :
^
In file included from main.cpp:24:0:
lambda.hpp:160:16: error: struct tml::impl::evaluate_impl<tml::impl::multi_lambda<BODY, VARIABLES ...>, ARGS ...>
struct evaluate_impl<multi_lambda<BODY,VARIABLES...>,ARGS...> :
我正在使用GCC4.8.2
首先,您应该真正了解SSCCE的真正含义,尤其是"完整">部分。 另外,简短。也就是说,我试图创建一个SSCCE,它似乎重现了您的问题,请参阅我的答案末尾。查看您收到的错误消息,似乎您的第三次专用化的真实代码看起来更像
template<template<typename...> class F ,
typename... PLACEHOLDERS ,
typename ARG ,
typename... ARGS>
struct evaluate_impl<F<PLACEHOLDERS...>,ARG,ARGS...>
: public F<tml::eval<ARG,ARGS>...>
{};
请注意对ARG
的附加明确提及,这似乎是多余的,可能会导致您情况中的歧义。如果将其替换为
template<template<typename...> class F ,
typename... PLACEHOLDERS ,
typename... ARGS>
struct evaluate_impl<F<PLACEHOLDERS...>,ARGS...>
: public F<tml::eval<ARGS>...>
{};
问题可能会消失。
最后,这里有一个SSCCE,我曾经遇到过与你类似的错误。
更新:使用下面评论中的SSCCE,只需在foo
时禁用F
的专业化即可解决这种情况。条件如下所示:
typename std::enable_if<!std::is_same<F<>,foo<>>::value>::type
或查看基于您的 SSCCE 的完整实时示例。这样,您可能还可以重新添加ARG
因为这两个专业现在应该是相互排斥的。
看起来在你的multi_lambda中你正在声明带有变量的模板......在最后。
template<typename BODY , typename... VARIABLES>
struct multi_lambda
{
//...
};
也许在您的使用中,您应该使用:
using lambda_2 = tml::multi_lambda<F<_1,_1,_2,_2>,_1,_2>;
- 嵌套模板显式专用化
- 构造函数 (g++) 的显式模板专用化
- GCC 和非命名空间范围内的显式专用化
- 如果专用化已经隐式实例化,它是否隐式实例化?
- 对类模板成员的显式专用化的约束
- 在 Xcode 中实例化后的显式专用化
- 模板变量的显式专用化
- 使用类型特征的部分类专用化
- 函数模板(它是类模板的成员)的显式专用化会产生"partial specialization is not allowed"错误,为什么?
- 将 CRTP 与部分类专用化结合使用?
- 函数模板中的显式模板专用化不起作用
- 我的C++模板失败:非模板结构的显式专用化
- 显式专用化 - 模板 ID 与任何模板声明都不匹配
- C++隐式/显式模板方法专用化问题
- 是否可以在没有显式专用化的情况下调用可变参数模板函数?
- 变量模板的显式专用化
- C++属于模板类的成员模板的显式专用化
- 非命名空间范围内的显式专用化不会在 GCC 中编译
- 错误 C2908:显式专用化; 已被实例化
- 针对多种类型的部分类模板专用化