如何强制编译器推断类模板参数?

时间:2019-04-21 22:40:30

标签: c++ templates type-deduction

我是c ++模板的新手,但我(很失败)试图迫使编译器在初始化时推断出模板typename参数。

这是我的代码。

template <typename T>
class C
{
public:
        T a;
        C(T a) : a(a) {}
        C(const C<T>& other) : a(other.a) {}
};

int main()
{
        C<int> x(1);
        C y{ x };

        return 0;
}

由g ++编译的此代码会导致错误。

test.cpp:13:11: error: missing template arguments before ‘y’
         C y{ x };
           ^

我想保持这种语法-仅使用C,而无需明确指定模板参数。

我尝试使用演绎指南,但这只会引发另一个错误。

template <typename T> C(const C<T>& other) -> C<T>;

当我将此行插入C类的定义下方时,我明白了。

test.cpp:10:51: error: expected constructor, destructor, or type conversion before ‘;’ token
 template <typename T> C(const C<T>& other) -> C<T>;
                                                   ^

当我将此行放入C类定义(在顶部)中时,发生了另一个错误。

C(const C<T>& other) -> C<T>;
test.cpp:4:26: error: ‘C’ function with trailing return type not declared with ‘auto’ type specifier
  C(const C<T>& other) -> C<T>;
                          ^~~~

在两种情况下,仍然都存在首次提及错误。

谢谢!

1 个答案:

答案 0 :(得分:1)

在C ++ 11中,仅模板函数可以推导其模板参数。例如,给定:

void f(std::pair<int, char> &s);

,可以这样称呼它

f(std::make_pair(5, 'c'));

,因为std::make_pair是一个函数,可以为函数推导模板参数。

但是,成对使用它是非法的:

f(std::pair(5, 'c'));

,因为类模板没有模板参数推导。此问题已在C ++ 17中修复,使std::make_pair有点过时。

有关新类模板参数推导的更多信息,请参见here

为了解决您的问题,在使用C ++ 11使用gcc进行编译时,我已经遇到了相同的错误。代码使用C ++ 17编译时没有错误(对于gcc,可以通过传递-std=c++17参数来完成)。