加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

c – 检查模板是否具有给定类型的特化

发布时间:2020-12-16 09:46:35 所属栏目:百科 来源:网络整理
导读:假设我们有一个这样的元函数: templatetypename LHS,typename RHSstruct add; 我们有一组已知类型的专业化.例如,整体包装: templatetypename T1,T1 VALUE1,typename T2,T2 VALUE2struct addstd::integral_constantT1,VALUE1,std::integral_constantT2,VALU
假设我们有一个这样的元函数:

template<typename LHS,typename RHS>
struct add;

我们有一组已知类型的专业化.例如,整体包装:

template<typename T1,T1 VALUE1,typename T2,T2 VALUE2>
struct add<std::integral_constant<T1,VALUE1>,std::integral_constant<T2,VALUE2> : public std::integral_constant<std::common_type_t<T1,T2>,VALUE1+VALUE2> {}; //Take advantage of C++14 helpers

几天前,我们创建了一个新类,我们专门为这个类添加了:

template<typename X,typename Y>
struct point_2d
{
    using x = X;
    using y = Y;
};

template<typename X1,typename Y1,typename X2,typename Y2>
struct add<point_2d<X1,Y1>,ppoint_2d<X2,Y2>> : public point_2d<add<X1,X2>,add<Y1,Y2>> {};

如您所见,我使用add metafuntion来执行坐标的添加.因此,任何具有添加元函数专门化的类型都可以用作point_2d坐标.

我的问题是:有没有办法检查模板是否具有给定类型作为参数的特化?

像这样的东西:

template<template<typename...> class TEMPLATE,typename... Ts>
struct has_specialization;

template<template<typename...> class TEMPLATE,typename... Ts>
struct has_specialization<TEMPLATE<Ts...> /* TEMPLATE<Ts...> exists */,Ts...> : public std::true_type {};

template<template<typename...> class TEMPLATE,typename... Ts>
struct has_specialization<TEMPLATE<Ts...> /* TEMPLATE<Ts...> doesn't exists */,Ts...> : public std::false_type {};

解决方法

@BartekBanachewicz the idea is to provide user friendly compiler errors through asserts.
If i can check (Following with the point_2d example) if the coordinates passed are addable
(That is,a specialization of add metafuncion is or not defined for that coordinates),
I can provide more readable errors like “That coordinates are not addable. point_2d needs
addable coordinates for point addition”,instead of the common horrible template instantation errors.

鉴于这是你的问题的动机,这不是一个
更简单直接的解决方案:

#include <type_traits>

template<typename LHS,typename RHS>
using cannot_add =  std::integral_constant<bool,std::is_same<LHS,LHS>::value>;

template<typename LHS,typename RHS>
struct add
{
    /* Assert some expressively named condition that is always false 
        but is dependent on a template parameter.
    */
    static_assert(!cannot_add<LHS,RHS>::value,"Types are not addable. Need specialization");
};

template<typename T1,VALUE2>> 
: public std::integral_constant<
    typename std::common_type<T1,T2>::type,VALUE1+VALUE2
> {};  

int main()
{
    add<std::integral_constant<int,1>,std::integral_constant<int,2>> x; // Line 25 
    add<float,float> y; // Line 26
    return 0;
}

GCC 4.8.1诊断:

main.cpp:26:19:   required from here
main.cpp:12:2: error: static assertion failed: Types are not addable. Need specialization

Clang 3.3诊断:

main.cpp:12:2: error: static_assert failed "Types are not addable. Need specialization"
    static_assert(!cannot_add<LHS,^             ~~~~~~~~~~~~~~~~~~~~~~~~~~~
main.cpp:26:19: note: in instantiation of template class 'add<float,float>' requested here
    add<float,float> y; // Line 26
                 ^

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读