我有这段代码(从更复杂的版本中简化):
template <class... A1> class Test {
public:
template <class... A2> void print (void(*function)(A2...,A1...)) {
}
};
void test_print (int a, float b, double c) {
}
int main () {
Test<float,double> test;
test.print<int> (&test_print);
}
如果我使用g++ -std=c++0x filename.cpp
在GCC 4.6.3上编译它,那么编译就好了但是在使用clang++ -std=c++0x filename.cpp
的clang 3.0上它会引发以下错误:
filename.cpp:14:10: error: no matching member function for call to 'print'
test.print<int> (&test_print);
~~~~~^~~~~~~~~~
filename.cpp:3:33: note: candidate template ignored: failed template argument deduction
template <class... A2> void print (void(*function)(A2...,A1...)) {
^
1 error generated.
在GCC 4.7.2上也有一个错误:
filename.cpp: In function 'int main()':
filename.cpp:14:33: error: no matching function for call to 'Test<float, double>::print(void (*)(int, float, double))'
filename.cpp:14:33: note: candidate is:
filename.cpp:3:33: note: template<class ... A2> void Test::print(void (*)(A2 ..., A1 ...)) [with A2 = {A2 ...}; A1 = {float, double}]
filename.cpp:3:33: note: template argument deduction/substitution failed:
filename.cpp:14:33: note: mismatched types 'float' and 'int'
现在问题是:为什么失败或我做错了什么?
答案 0 :(得分:2)
作为解析http://www.open-std.org/jtc1/sc22/wg21/docs/cwg_defects.html#1399的一部分,决定不可推导的参数包不会参与演绎,因此将明确指定的参数保留在参数列表中,但不会干扰后续参数的匹配后续的功能参数。但是,在比较类型时,这些添加的规则不一定适用于推论(这是一组参数的推导与函数调用的参数的推导)。
我认为该决议应适用于您的情况,并将其视为一个规范尚不完全清楚的领域。您可能希望向相应的编制者发送错误报告,他们可能希望向WG21发送问题报告,要求澄清这一点(尽管我不确定我会说这是一个规范性的规范问题 - 标准在其他地方说,不可扣除的背景不参与扣除)。
注意,A1
是封闭类模板的参数(因此在实例化类时成为正常参数),您可以省略它们并保持与模板段扣除相关的含义
class Test {
public:
template <class... A2> void print (A2..., float, double) {
}
};
int main () {
Test test;
test.print<int> (1, 1.0f, 1.0); // works
}