定义类模板的一般形式是:
template <类型名 参数名1,类型名参数名2,…>
class 类名
{
类声明体
};
例如,template <class T>
class Smemory
{…
public:
void mput(T x);
…
}
表示定义一个名为Smemory的类模板,其中带类型参数T。
在类模板的外部定义类成员函数的一般形式是:
template <类型名 参数名1,类型名参数名2,…>
函数返回值类型 类名<参数名 1 参数名 2,…>::成员函数名(形参表)
{
函数体
}
例如:template <class T>
void Smemory<T>::mput(T x)
{…}
表示定义一个类模板Smemory的成员函数,函数名为mput,形参x的类型是T,函数无返回值。
类模板是一个类家族的抽象,它只是对类的描述,编译程序不为类模板(包括成员函数定义)创建程序代码,但是通过对类模板的实例化可以生成一个具体的类以及该具体类的对象。
与函数模板不同的是:函数模板的实例化是由编译程序在处理函数调用时自动完成的,而类模板的实例化必须由程序员在程序中显式地指定,
其实例化的一般形式是:
类名 <数据类型 1(或数据),数据类型 2(或数据)…> 对象名
例如,Smemory<int> mol;
表示将类模板Smemory的类型参数T全部替换成int 型,从而创建一个具体的类,并生成该具体类的一个对象mol。
下面举例说明类模板的定义和使用方法。
例9.3.1:类模板的定义和使用。
#include <iostream.h>
#include <conio.h>
const int SIZE=8;
template <class T>
class Smemory { //定义类模板Smemory
T data[SIZE]; //类型为T,长度为SIZE的数组data[]为数据成员
int count;
public:
Smemory( ){ count=0; }
void mput(T x); //mput()函数的参数x的类型是T
T mget( ); //声明返回值类型为T的成员函数mget()
};
template <class T>
void Smemory<T>::mput(T x) //定义成员函数mput(),函数的参数类型为T,该函数用于为数据成员 data数组的各个元素赋值
{
if(count==8) { cout<<"Memory is full"; return; }
data[count]=x;
count++;
}
template <class T>
T Smemory<T>::mget( ) //定义成员函数mget(),函数的返回类型为T,该函数用于取出数据成员 data数组的各个元素
{
if(count==0) { cout<<"Memory is empty"; return 0; }
count--;
return data[count];
}
void main( )
{
Smemory<int> mo1;
int i; char ch='A';//将Smemory实例化,并创建对象mo1
Smemory<char> mo2; //将Smemory实例化,并创建对象mo2
for(i=0; i<8;i++)
{
mo1.mput(i); //调用成员函数mput()
mo2.mput(ch); ch++; //调用成员函数mput()
}
cout<<"Get mo1 => ";
for(i=0;i<8;i++)
cout<<mo1.mget( ); //调用成员函数mget()
cout<<"\nGet mo2 => ";
for(i=0;i<8;i++)
cout<<mo2.mget( ); //调用成员函数mget()
getch();
}
程序的运行结果是:
Get mo1=> 76543210
Get mo2=> HGFEDCBA
说明:类模板Smemory带一个类型参数T,T是代表数据类型的参数,在类型参数前面必须加关键字class,用class表示类型参数的类型。在实例化时,对应类型参数T必须是具体的数据类型名,这里建立第一个对象时数据类型名是int,表示将类模板Smemory中的所有类型参数T都替换成int型, 从而创建一个具体的类及其对象mo1;在建立第二个对象时数据类型名是char,表示将模板Smemory中的所有类型参数T都替换成char,从而创建一个具体的类及其对象mo2。
来源:http://jpk.sdju.edu.cn/cplus/kejian/content/chapter9/chapter9_3_2.htm