我看到了一件有趣的事情,但不明白为什么。
template<class dataType>
Class A
{
AFnc();
}
template<> A<int>::AFnc() { }
Run Code Online (Sandbox Code Playgroud)
仅使用专用模板会产生错误,指出同一函数的多重定义。它说它是在同一个地方生成的。但如果我添加
template<class dataType>
A<dataType>::AFnc()
{
}
Run Code Online (Sandbox Code Playgroud)
然后它摆脱了错误。
为什么 ?有人可以解释这种行为。
(你需要清理你的语法。我假设实际的代码没有所有这些语法错误。)
模板函数的显式特化不再是模板,因为它不再依赖于任何模板参数。从单一定义规则 (ODR) 的角度来看,它是一个“普通”功能。而且,作为一个“普通”函数,它必须在头文件中声明,并且在某些实现文件中只定义一次。您显然在头文件中定义了您的专业化,如果头文件被包含在多个翻译单元中(例如您的“多重定义”错误),这会导致 ODR 违规。
在您的示例中,template<> void A<int>::AFnc()
(我添加void
为返回类型)不再是模板。这意味着这个定义
template<> void A<int>::AFnc() { }
Run Code Online (Sandbox Code Playgroud)
必须从头文件移动到某个实现文件。同时,在头文件中,你必须为这个函数保留一个非定义声明
template<> void A<int>::AFnc(); // <- note, no function body
Run Code Online (Sandbox Code Playgroud)
让编译器知道存在这样的专业化。
一般来说,记住一个简单的规则:如果你的函数模板仍然依赖于一些未指定的模板参数,它就是一个真正的模板,它必须在头文件中定义。但是一旦你“修复”了所有参数(通过显式专业化),它就不再是一个模板。它变成了一个普通的函数,必须在头文件中声明,并且在某个实现文件中只定义一次。
PS 以上适用于非内联函数。内联函数可以(并且通常应该)在头文件中定义。
PPS 同样的逻辑适用于模板类的静态数据成员的显式特化。
归档时间: |
|
查看次数: |
1119 次 |
最近记录: |