文章目录
- 泛型编程
- 模板
- 函数模板
- 概念
- 原理
- 函数模板的实例化
- 类模板
泛型编程
我们在实现交换函数的时候,只能实现一个数据类型的交换函数,想要在C++中完成对应类型数据的交换一种方法是使用函数重载,就像下面这样
void Swap(int& left, int& right)
{
int temp = left;
left = right;
right = temp;
}
void Swap(double& left, double& right)
{
double temp = left;
left = right;
right = temp;
}
void Swap(char& left, char& right)
{
char temp = left;
left = right;
right= temp;
}
这样写就显得我们不太聪明的样子,这样写的缺点是:重载的函数仅仅是类型不同,代码复用率较低,只要有新类型出现时,就需要用户自己增加对应的函数。代码的可维护性比较低,一个出错可能所有的重载出错。
针对这些问题,C++中提供了模板的语法
模板
泛型编程:编写与类型无关的通用代码,是代码复用的一种手段。模板是泛型编程的基础。后面我们将会大量运用泛型编程。
模板可以分为函数模板和类模板
函数模板
我们来看一下这个代码:
template <typename T>
void Swap(T& a, T& b)
{
T temp = a;
a = b;
b = temp;
}
上面的代码运用了函数模板的概念,template是C++中提供函数模板参数的一个关键字,上面就是一个交换函数的函数模板,该函数可以适应很多类型的数据交换。
概念
函数模板代表了一个函数家族,该函数模板与类型无关,在使用时被参数化,更具实参类型产生函数的特定类型版本。
格式:template <typename T1,typename T2,……typename Tn>
返回值类型 函数名(参数列表) {}
注意:typename是用来定义模板参数关键字,也可以使用class(切记:不能使用struct代替class)。
原理
函数模板只是一个模板,它本身不是函数,是编译器使用方式产生特定具体类型函数的模具。所以其实模板就是将本来我们做的重复的事情交给了编译器。
在编译阶段,编译器需要根据传入的实参类型来推演生成对应类型的函数。
函数模板的实例化
template <typename T>
T Add(const T& left, const T& right)
{
return left + right;
}
int main()
{
int a1 = 1, a2 = 1;
double d1 = 1.1, d2 = 1.2;
Add(a1, a2);
Add(d1, d2);
//a1和d1不是相同类型的数据,编译器在推演实际的参数是什么类型的时候,
//会发生二义性的问题,也就是说函数模板参数只有一个编译器不知道是int还是double
//Add(a1, d1);
//我们可以这样处理
//这时候我们需要强制类型转换。但是这里还有一个问题,强制类型转换会产生临时变量
//编译器把这个临时变量传给函数的时候,函数的参数列表必须有const修饰,这是由于
//临时变量具有常性,如果不加const就会造成权限放大的问题,这时候加上const就是权限平移
//不会出错
//1.用户强制转换,2.使用显式实例化(在<>中指定模板参数的实际类型)
Add(a1, (int)d1);
Add<int>(a2, d2);
}
也可以这样写,模板参数设置为两个
//通用加法函数
//这样写就不会发生两个类型不同,编译器不知道实例化成哪个的问题了
//但是这样会使精确度下降
template <typename T1,typename T2>
T1 Add(T1& left, T2& right)
{
return left + right;
}
int main()
{
int a = 1;
double b = 1.1;
cout << Add(a, b) << endl;
}
类模板
定义格式:
template<class T1,class T2,……,class Tn>
class 类模板名
{
//成员函数
}
实现一个模板栈
template <typename T1>
//模板类
class Stack
{
public:
Stack(int capacity = 3)
{
_array = new T1[capacity];
_capacity = capacity;
_top = 0;
}
void CheckCapacity()
{
if (_top == _capacity)
{
T1* temp = (T1*)realloc(_array, sizeof(T1) * _capacity * 2);
if(NULL == temp)
{
perror("realloc failed!\n");
return;
}
_array = temp;
_capacity *= 2;
cout << "扩容成功!" << endl;
}
}
void PushStack(T1 x)
{
CheckCapacity();
_array[_top] = x;
_top++;
}
void PopStack()
{
if (_top > 0)
{
_top--;
}
else
{
cout << "退栈失败,栈已为空!" << endl;
return;
}
}
bool EmptyStack()
{
return _top == 0;
}
T1 StackTop()
{
return _array[_top - 1];
}
~Stack()
{
delete[] _array;
_array = NULL;
_capacity = _top = 0;
}
private:
T1* _array;
int _capacity;
int _top;
};
int main()
{
Stack<int> s1;
s1.PushStack(1);
s1.PushStack(2);
s1.PushStack(3);
s1.PushStack(4);
s1.PushStack(5);
s1.PushStack(6);
s1.PushStack(7);
while (!s1.EmptyStack())
{
cout << s1.StackTop() << endl;
s1.PopStack();
}
/*Stack<double> s2;
Stack<short> s3;*/
return 0;
}
需要注意:Stack 为类名,Stack<int> 才是类型
在类外定义函数时应该这么写
Stack<T> :: Stack() //在类外定义构造函数