当类模版遇到继承时,需要注意以下几点:
- 当子类继承的父类是一个类模版时,子类在声明的时候,要指定出父类中T的类型
- 如果不指定,编译器无法给子类分配内存
- 如果想灵活指定出父类中T的类型,子类也需变为类模版
因为父类中的一些东西需要继承到子类中,父类中有的成员,子类中也必须有,但是子类也不是一个模版,因此一个子类对象到底占据多少个内存空间,怎么算?算不出来,是因为我确定不出来这个继承的这个东西到底占据了多大的空间,因此发生了继承的时候,我必须指明父类中的T 究竟是什么样的数据类型,才能继承该子类。
cpp
#include <iostream>
//类模版与继承
template<typename T>
class Base{
T m;
};
class Son:public Base<int>{//必须要知道父类中T的数据类型才能继承给子类
};
void test01()
{
Son s1;
}
int main() {
test01();
return 0;
}
但是如果想灵活的指定父类中T的数据类型,子类也需要编程类模版。
cpp
//如果想灵活的指定父类中T的数据类型,子类也需要编程类模版
template <typename T1,typename T2>
class Son2:public Base<T2>
{
T1 obj;
T2 obj2;
};
这里的int和char传给了T1和T2
cpp
//如果想灵活的指定父类中T的数据类型,子类也需要编程类模版
template <typename T1,typename T2>
class Son2:public Base<T2>
{
T1 obj;
};
void test02()
{
Son2<int,char>S2;
}
这里的T1是obj的数据类型,他现在是一个整型int,这个T2给了父类,因此这个T2是m的数据类型,其中这个数据类型是char类型。这样一步一步的成立之后,再继承下来之后,这样他们就能拿到一个父类中的m。
cpp
//如果想灵活的指定父类中T的数据类型,子类也需要编程类模版
template <typename T1,typename T2>
class Son2:public Base<T2>
{
//写一个构造函数来看一下T1和T2的数据类型
public:
Son2(){
std::cout<< "T1 type: " << typeid(T1).name() << std::endl;
std::cout<< "T2 type: " << typeid(T2).name() << std::endl;
}
T1 obj;
};
void test02()
{
Son2<int,char>S2;
}
int main() {
test02();
return 0;
}
这里再运行Son2的时候,其一定会调用自己的构造函数。
最终的结果显示其一个为int型,一个为char型。
总结:父类是一个类模版,子类在继承的时候需要指定出父类中T的数据类型。