定义外部成员模板功能

时间:2015-07-18 16:36:33

标签: c++ templates language-lawyer member-functions

考虑这个(最小化)示例:

template <typename Descriptor>
class hash_table
{
public:
  typedef int value_type;

  template <typename Argument, int Callback (value_type *, Argument)>
  void traverse (Argument);

  template <int Callback (value_type *)>
  void traverse_void ();
};

我定义了一个类模板,它具有带非类型参数的模板成员函数。请注意,Callback取决于value_type typedef。现在我想自己定义函数:

template <typename Descriptor>
template <typename Argument, int Callback (typename hash_table <Descriptor>::value_type *, Argument)>
void hash_table <Descriptor>::traverse (Argument) {}

template <typename Descriptor>
template <int Callback (typename hash_table <Descriptor>::value_type *)>
void hash_table <Descriptor>::traverse_void () {}

我从编译器中得到了不一致的错误。结果不依赖于选项,指定C ++标准的版本(即C ++ 98,C ++ 11和C ++ 14的版本),但取决于编译器。

GCC 6.0.0(最近的主干,以及其他几个版本)接受此代码。

Clang 3.7.0(最近的主干)出现以下错误:

test.cc:18:31: error: out-of-line definition of 'traverse_void' does not match any declaration in 'hash_table<Descriptor>'
void hash_table <Descriptor>::traverse_void() {}
                              ^~~~~~~~~~~~~
1 error generated.
EDG(英特尔C ++编译器v.15.0.3)给出了两个错误:

test.cc(15): error: declaration is incompatible with function template "void hash_table<Descriptor>::traverse<Argument,Callback>(Argument)" (declared at line 7)
  void hash_table <Descriptor>::traverse (Argument) {}
                                ^

test.cc(19): error: declaration is incompatible with function template "void hash_table<Descriptor>::traverse_void<Callback>()" (declared at line 10)
  void hash_table <Descriptor>::traverse_void () {}
                                ^
compilation aborted for test.cc (code 2)

预期的行为是什么(根据标准)?如果代码错误,我该如何修复函数定义?

1 个答案:

答案 0 :(得分:1)

你的代码看起来很好,我认为没有理由不编译。可能是禁止它的编译器的错误。但是,因为[temp.param]:

  

类型为“T数组”或“函数返回T”的非类型模板参数被调整为类型   “指向T的指针”或“指向函数返回T的指针”。

您可以简单地将Callback切换为指向函数的指针。这种行为在所有方面都是相同的,这个代码也可以在clang上编译。