clang ++在使用CRTP时不接受使用模板模板参数

时间:2013-07-16 21:37:32

标签: c++ templates c++11 clang crtp

在使用带有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 ++更晚了!

1 个答案:

答案 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 >
//                            ^^