在使用带有CRTP的模板模板参数时,尝试在派生初始化列表中调用基类构造函数时出现编译错误。
可以使用以下代码段复制问题:
template <template<class> class Derived, class T>
struct base
{
};
template <class T>
struct derived : public base<derived, T>
{
derived()
: base<derived, T>()
{ }
};
违规错误消息:
bug.cpp:10:16: error: template argument for template template parameter must be a class template or type alias template
: base<derived, T>()
^
bug.cpp:10:11: error: expected class member or base class name
: base<derived, T>()
^
bug.cpp:10:11: error: expected '{' or ','
3 errors generated.
这个问题似乎只发生在clang(3.4)上,而不是g ++(4.8,4.7,4.6)。我正在使用-std = c ++ 11进行编译。
这是我第一次将CRTP与模板模板参数一起使用。我这样做是否正确,这是clang ++的问题吗?
我已经开始相信clang ++错误消息比g ++更晚了!
答案 0 :(得分:17)
您的代码是合法的。
从C ++ 11标准,第14.6.1节:
与普通(非模板)类一样,类模板具有注入类名(第9节)。 inject-class-name可以用作模板名称或类型名称。 当使用时使用template-argument-list,作为模板参数的模板参数,或者作为详细类型指定器中的最终标识符朋友类模板声明,它指的是类模板本身。
您的clang
版本似乎仍在实施旧规则。根据您的其他评论,它可能只在 ctor-initializer-list 中执行。
用户David Rodríguez - dribeas为尚未完全实现C ++ 11注入类名称规则的编译器提供了一种解决方法。使用不合格的类的任何名称,例如:
derived()
: base< ::derived, T >()
// ^^ qualified with global namespace
{ }
某些编译器也可能在继承列表中要求这样做:
template <class T>
struct derived : public base< ::derived, T >
// ^^