我想为模板类做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>'|
问题是什么,这样的事情会有什么好处?
答案 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
参数。实际上,编译器只是忽略它们
例如在原始代码中,T
和TDIM
在以下行中被忽略:
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> > {
};