如何有条件地实例化具有多个模板参数的模板类?

时间:2019-05-03 07:49:17

标签: c++ class templates c++17 sfinae

我关注了这篇文章:Class template SFINAE有条件地实例化模板类。

这对于只有一个模板参数的类非常有效,如上面的链接所示。

但是,我有两个(模板)参数,我想做一些SFINE检查。 以下是我的代码的最小示例。

#include <type_traits>
#include <string>

template<class T, class U, class R> using arithmetic_types =  std::enable_if_t<
    std::is_arithmetic_v<T> &&
    std::is_arithmetic_v<U>,
    R
>;

template<class T, class U, class Enable = void> class MyClass;
template<class T, class U, arithmetic_types<T, U, void>> 
class MyClass {
public:
    MyClass() = default;
};

int main()
{
    MyClass<int, int> o;          // should work
    MyClass<int, double> o1;      // should work
    MyClass<int, std::string> o2; // should be a complier error
    return 0;
}

上面给了我错误信息:https://godbolt.org/z/BEWJMp

error C3855: 'MyClass': template parameter 'Enable' is incompatible with the declaration
error C2079: 'o' uses undefined class 'MyClass'
error C2079: 'o1' uses undefined class 'MyClass'
error C2079: 'o2' uses undefined class 'MyClass'

很遗憾,我无法理解错误消息(error C3855:)。

为什么我不能执行上面链接中所示的相同原理来获取更多模板参数

什么是最佳解决方案

1 个答案:

答案 0 :(得分:6)

问题出在MyClass的模板特化中。只能在两个类TU上对专业化进行参数化,测试应放在声明中,如下例所示。

#include <string>
#include <type_traits>

template <class T, class U, class R>
using arithmetic_types = std::enable_if_t<
    std::is_arithmetic_v<T> && std::is_arithmetic_v<U>, R>;

template <class T, class U, class Enable = void>
class MyClass;

template <class T, class U> //<- Remove the test from here
class MyClass<T, U, arithmetic_types<T, U, void>> //<- Put the test here.
{
public:
  MyClass() = default;
};

int main()
{
  MyClass<int, int> o;          // should work
  MyClass<int, double> o1;      // should work
  MyClass<int, std::string> o2; // should be a complier error
  return 0;
}

演示:https://godbolt.org/z/xTnwo9