使用C ++ 11 <type_traits> </type_traits>模板参数类型扣除失败

时间:2013-06-07 14:14:57

标签: c++ c++11 typetraits template-deduction

我正在尝试了解如何使用C ++(11)<type_traits>

这是我的琐碎测试程序

#include <type_traits>

template<class U, class S>
inline U add(typename std::enable_if<std::is_unsigned<U>::value,U>::type a,
             typename std::enable_if<std::is_signed  <S>::value,S>::type b)
{
    return a + b;
}

int main(int argc, const char * argv[], const char * envp[])
{
    unsigned int ui;
    int i;
    auto a = add(ui, i);
    return 0;
}

使用GCC 4.8.1编译时,错误为

/home/per/f.cpp: In function ‘int main(int, const char**, const char**)’:
/home/per/f.cpp:15:23: error: no matching function for call to ‘add(unsigned int&, int&)’
     auto a = add(ui, i);
                       ^
/home/per/f.cpp:15:23: note: candidate is:
/home/per/f.cpp:5:10: note: template<class U, class S> U add(typename std::enable_if<std::is_unsigned<U>::value, U>::type, typename std::enable_if<std::is_signed<S>::value, S>::type)
 inline U add(typename std::enable_if<std::is_unsigned<U>::value,U>::type a,
          ^
/home/per/f.cpp:5:10: note:   template argument deduction/substitution failed:
/home/per/f.cpp:15:23: note:   couldn't deduce template parameter ‘U’
     auto a = add(ui, i);
                       ^

我不知道为什么GCC无法推断出模板参数U。有人知道我的代码丢失了什么信息,那就是我在C ++ 11中编写一个程序,它将无符号整数类型作为第一个参数并将有符号整数类型作为第二个参数?

5 个答案:

答案 0 :(得分:13)

typename std::enable_if<std::is_unsigned<U>::value,U>::type不是可推断的背景。为了从中推导U,编译器需要能够应用std::enable_if的反向操作。它看起来并不太难,这是真的,但那是因为你在谈论像enable_if这样的简单事情。要求每一个特性都是不可能的,所以C ++只是玩得很​​酷而且不会产生任何奇怪的规则异常:它一般不可推导,它在这一点上是不可推断的。

你可以这样做:

template<class U, class S,
         EnableIf<std::is_unsigned<U>, std::is_signed<S>>...>
         // see http://flamingdangerzone.com/cxx11/2012/06/01/almost-static-if.html
U add(U a, S b)

或者在不正确支持该样式的编译器中,您只需添加一个额外的默认参数:

template<class U, class S>
U add(U a, S b,
      typename std::enable_if<std::is_unsigned<U>::value
          && std::is_signed<S>::value,void>::type* = nullptr)

...或搞乱返回类型。

template<class U, class S>
typename std::enable_if<std::is_unsigned<U>::value
    && std::is_signed<S>::value,U>::type
add(U a, S b)

答案 1 :(得分:7)

您没有给编译器推断US的机会。您可以按如下方式重写函数,并在模板参数列表中移动SFINAE检查:

template<class U, class S,
    typename std::enable_if<std::is_unsigned<U>::value &&
                            std::is_signed  <S>::value
        >::type* = nullptr>
inline U add(U a, S b)
{
    return a + b;
}

这是live example

答案 2 :(得分:3)

您首先必须在之前推断类型,然后才能推断出类型!

应该是:

template <typename U, typename S>
typename std::enable_if<std::is_unsigned<U>::value &&
                        std::is_signed<S>::value>, U>::type
add(U u, S s)
{
    // ...
}

答案 3 :(得分:2)

无法从“嵌套的typedef”表达式中推导出模板参数。也就是说,可以从U推断出some_template<U>,但不能从some_template<U>::type推断出来。

编译器不可能枚举some_template的所有(无限!)实例化,并查看嵌套的typedef中的哪一个等于实际的参数类型。

答案 4 :(得分:2)

尝试:

template<class U, class S>
typename std::enable_if<std::is_unsigned<U>::value && std::is_signed<S>,U>::type  
add(U a , S b)
{
    return a + b;
}