一,类模板语法
类模板与函数模板的定义和使用类似,即:有时,有两个或多个类,其功能是相同的,仅仅是数据类型不同。
类模板用于实现所需数据的类型参数化
类模板在表示如数组、表、图等数据结构显得特别重要
这些数据结构的表示和算法不受所包含的元素类型的影响。
二、单个类模板语法
例子
template<typename T>
class A
{
public:
A(T t)
{
this->t = t;
}
T &getT()
{
return t;
}
public:
T t;
}
void main()
{
//模板中如果使用了构造函数,则遵守以前的类的构造函数的调用规则
A<int> a(100);
a.getT();
printAA(a);
return ;
}
三、类模板
模板类派生时,需要具体化模板类,C++编译器需要知道 父类的数据类型具体是什么样子的
要知道父类所占的内存大小是多少,只有数据类型固定下来,才知道如何分配内存。
class B : public A<int>
四、从模板类派生模板类
typename <typename T
>
class C : public A<T>
{
public:
C(T c, T a) : A<T>(a)
{
this->c = c;
}
void printf()
{
cout<<"c: "<<c<<endl;
}
protect:
T c;
}
在实际项目中,函数的声明与实现一般不在一起,如果想要在类模板中实现函数,举例:
template <typename T>
void A<int> :: printA()
{
cout<<"a: "<<a<<endl;
}
五、所有的类模板函数写在类的外部,在一个cpp中
友元函数:用友元函数重载<<>>(左移右移)
friend ostream& operator<< <T>(ostream &out , Complex<T>&c3)
友元函数:友元函数不是实现函数重载,即非左移右移<< >>
1、需要在类前增加 类的前置声明 函数的前置声明ypename
template<typename T>
class Complex;
template <typename T>
Complex<T> mySub ( Complex<T> &c1, Complex<T> &c2);
2、类的内部声明 必须写成:
friend Complex<T> mySub<T> (Complex<T>&c1 , Complex<T>&c2);
3、友元函数实现 必须写成:
template<typename T>
Complex<T> mySub(Complex<T>&c1, Complex<T>&c2)
{
Complex<T> tmp(c1.a - c2.a , c1.b - c2.b);
return tmp;
}
4、友元函数调用 必须写成
Complex<int> c4 = mySub<int> (c1 , c2);
cout<<c4;
结论:友元函数只用来进行 左移 右移操作符重载
六、可以这样声明和使用类模板
1、先写出一个实际的类,由于其语义明确,含义清楚,一般不会出错
2、将此类中准备改变的类型名(如 int要改变为 float 或 char )改用一个自己指定的虚拟类型名
3、在类声明前面加入一行,格式为:
template <class 虚拟类型参数>
如:
template<class numtype>
{...}; //类体
4、用类模板定义对象时用以下形式:
类模板名 <实际类型名> 对象名;
类模板名 <实际类型名> 对象名(实参列表);
如:
Compare <int> cmp;
Compare <int> cmp(3,7);
5、如果在类模板外定义成员函数,应写成类模板形式:
template<class 虚拟类型参数>
函数类型 类模板名<虚拟类型参数> :: 成员函数名(函数形参列表){...}
七、类模板的几点说明
1、类模板的类型参数可以有一个或多个,每个类型前面都必须加 class ,如:
template <class T1 , class T2>
class someclass
{...}
在定义对象时分别带入实际的类型名,如:
someclass<int , double>obj;
2、和使用类一样,使用类模板时要注意其作用域,只能在其有效作用域内用它定义对象
3、模板可以有层次,一个类模板可以作为基类,派生类派生模板类。
八、类模板的static关键字
1、从类模板实例化的每个模板类有自己的类模板数据成员,该模板类的所有对象共享一个static数据成员
2、和非模板类的static数据成员一样,模板类的static数据成员也应该在文件范围定义和初始化。
3,每个模板类有自己的类模板的static数据成员副本