CRTP和模板模板?

时间:2012-08-13 05:11:57

标签: c++ templates crtp template-templates

我想为模板类做CRTP,我希望抽象基类知道派生类的模板参数。

我尝试了这个,但它不起作用:

template<template<class T, unsigned int TDIM> class TCRTP> class NAbstract
{
    T _data[TDIM];
};

template<typename T, unsigned int TDIM> class NArray : NAbstract< NArray<T, TDIM> >
{

};

template<typename T, unsigned int TDIM> class NVector : NAbstract< NVector<T, TDIM> >
{

};

g ++ 4.6.2告诉我:

main.cpp|5|error: 'T' does not name a type|
main.cpp|8|error: type/value mismatch at argument 1 in template parameter list for 'template<template<class T, unsigned int TDIM> class TCRTP> class NAbstract'|
main.cpp|8|error:   expected a class template, got 'NArray<T, TDIM>'|
main.cpp|13|error: type/value mismatch at argument 1 in template parameter list for 'template<template<class T, unsigned int TDIM> class TCRTP> class NAbstract'|
main.cpp|13|error:   expected a class template, got 'NVector<T, TDIM>'|

问题是什么,这样的事情会有什么好处?

3 个答案:

答案 0 :(得分:1)

下面演示修复错误后的代码:

template<class T, unsigned int TDIM, template<class, unsigned int> class TCRTP>
class NAbstract
{
  T _data[TDIM];
};

template<typename T, unsigned int TDIM>
class NArray : NAbstract< T, TDIM, NArray >
{
};

template<typename T, unsigned int TDIM>
class NVector : NAbstract< T, TDIM, NVector >
{
};

您无法使用template template参数。实际上,编译器只是忽略它们 例如在原始代码中,TTDIM在以下行中被忽略:

template<template<class T, unsigned int TDIM> class TCRTP> class NAbstract
                       ^^^              ^^^^  // <--- ignored

答案 1 :(得分:1)

一种解决方案是部分专门化模板:

template <typename> struct NAbstract;

template <typename T, unsigned int N>
struct NAbstract<NArray<T, N>>
{
    T _data[N];
    // ...
};

可替换地:

template <template <typename, unsigned int> class Container,
          typename T, unsigned int N>
struct NAbstract<Container<T, N>>
{
    T _data[N];
    // ...
};

答案 2 :(得分:0)

您可以使用特征来完成此任务。然后,您不需要NAbstract的模板模板参数。

template<class TCRTP>
class NAbstract {
    typename TCRTP::value_type data_[TCRTP::DIM];
};

template<typename T, unsigned int TDIM>
struct NTraits {
    typedef T value_type;
    enum { DIM = TDIM };
};

template<typename T, unsigned int TDIM>
struct NArray : NAbstract< NTraits<T, TDIM> > {
};

template<typename T, unsigned int TDIM>
struct NVector : NAbstract< NTraits<T, TDIM> > {
};