个人主页:Lei宝啊
愿所有美好如期而遇
目录
泛型编程
函数模板
类模板
泛型编程
我们先看一个代码:
看着是不是有点麻烦,我们有没有一种通用的办法,让编译器能够根据不同的类型自动生成不同的函数呢?有,就是模板。
泛型编程:
编写与类型无关的通用代码,是代码复用的一种手段。
而模板是泛型编程的基础
函数模板
我们先看代码:
template<typename T>
void Swap(T& a, T& b)
{
T tmep = a;
a = b;
b = temp;
}
函数模板概念 :
函数模板代表了一个函数家族,该函数模板与类型无关,在使用时被参数化,根据实参类型产生函数的特定类型版本。
我们以上面代码为例讲解:
首先,函数模板的格式:
template<typename T1,typename T2, ......>
返回值 函数名(参数列表){}
template 的意思就是模板。
typename是用来定义模板参数的关键字,也可以用class替代,但不可以使用struct。
T1,T2等就是编译器自己根据实参类型推导出来的,或者由我们显式实例化(后面说)。
接下来我们进行各种可能的实验。
错误原因是,编译器通过实参a将T推导为int类型,通过实参e将T推导为double类型,但是只有一个T,编译器无法确定他是int还是double,就会报错。
template<class T1, class T2>
void Swap(T1& a, T2& b)
{
double temp = a;
a = b;
b = temp;
}
多了这样一个重载就没问题了,因为我们多了一个模板参数,则T1推导为int,T2推导为double。
换成Add函数还有另一种解决办法:强制类型转换
template<class T>
T Add(T& a, T& b)
{
return a + b;
}
但是这里涉及引用的隐式类型转换,参照:C++入门,目录-引用,所以我们的Swap函数需要变点东西:
template<class T>
T Add(const T& a,const T& b)
{
return a + b;
}
ret结果为4。
去掉const,就会出现权限放大的问题,在e强制类型转换时产生临时变量,具有常性,而函数的参数是变量,就会报错。
还有一种解决办法,叫做显式实例化:
我们指定模板的模板参数,叫做显式实例化,编译器自己推导,叫做隐式实例化。
模板参数的匹配规则:
一个非模板函数可以和一个函数模板同时存在,而且该函数模板还可以被实例化为这个非模板函数。
#include <iostream>
using namespace std;
int add(int a, int b)
{
return a + b;
}
template<class T>
T add(T a, T b)
{
return a + b;
}
int main()
{
add(1, 2);
add<int>(1, 2);
return 0;
}
对于非模板函数和同名函数模板,如果调用函数时传参与非模板函数完全相同,那么就优先调用非模板函数,而不是调用函数模板实例化出来的函数;如果模板可以产生一个具有更好匹配的函数,那么将选择模板去实例化函数。
#include <iostream>
using namespace std;
int Add(int a, int b)
{
return a + b;
}
template<class T1, class T2>
T1 Add(T1 a, T2 b)
{
return a + b;
}
int main()
{
cout << Add(1, 2) << endl;
cout << Add(1.0, 2) << endl;
return 0;
}
我们进入调试时,第二个Add函数是模板实例化出来的函数。
类模板
类模板定义格式:
template<typename T1,typename T2, ......>
class 类模板名字{}
我们可以使用类模板写一个动态顺序表,而且通过类模板,我们更加能够体会到他的魅力以及他的强大,如果我们使用C语言来写,而我们的要求是写出多种数据类型的顺序表,用C语言需要重复拷贝多次去修改类型,代码会有很多重复,但是有了模板就不一样了。
template<class T>
class Vector
{
T* _Data;
int _size;
int _capacity;
public:
Vector(int capacity = 4)
:_size(0)
,_capacity(capacity)
,_Data(new T[_capacity])
{}
~Vector()
{
if (_Data)
delete[] _Data;
_Data = nullptr;
_size = _capacity = 0;
}
T& operator[](int pos)
{
if (pos >= 0)
{
assert(pos < _size && pos >= 0);
return _Data[pos];
}
}
};
各种类型的对象我们都可以实例化出来。
类模板的实例化
类模板实例化与函数模板实例化不同,需要显式写出模板参数
也许你有这样的疑问,是不是因为我没有传参数,所以类模板无法推导T是什么,但是我们可以看看class 类名,有参数列表吗?显然没有,所以才需要我们显式去传模板参数。
同时我们需要注意的是:
类模板并不是真正的类,就像模板函数一样,只有将他实例化后才是函数,才是真正的类。