模板是为了解决一个通用性问题而存在的,如果你写一个数据时不确定他是什么类型,可以通过模板来写这个类型。
C++中模板可以分为函数模板和类模板
模板的基本语法:
template <class T>
//函数或者类
要写函数模板就在下面写函数,写类模板就在下面写类就行了。
1.函数模板:
//函数模板
//声明一个模板,告诉编译器后面代码中紧跟着的T不要报错,T是一个通用的数据类型
template <typename T> //typename 可以替换成class
void mySwap(T &a, T &b)
{
T temp = a;
a = b;
b = temp;
}
void Test1()
{
int a = 10;
int b = 20;
//利用函数模板交换
//两种方式使用函数模板
//1.自动类型推导
//mySwap(a, b);
//2.显示指定类型
mySwap<int>(a, b);
}
在上述函数模板写的mySwap函数中,你不光能交换int数据的类型,而且可以交换char、float等等类型,但是不能交换自定义数据类型。
如果想处理自定义数据类型 例如类对象 需要自己重载操作符实现或者自己构造函数来实现,如下:
class Person
{
public:
Person(string name, int age)
{
m_Name = name;
m_Age = age;
}
int m_Age;
string m_Name;
};
//对比两个数据是否相等函数
template <class T>
bool myCompare(T &a, T &b)
{
if (a == b)
{
return true;
}
else
{
return false;
}
}
//利用具体化Person的版本来实现代码,具体化优先调用
template<> bool myCompare(Person &p1, Person &p2)
{
if (p1.m_Age == p2.m_Age && p1.m_Name == p2.m_Name)
{
return true;
}
else
{
return false;
}
}
2.类模板
类模板的语法和函数模板的语法类似,但是区别在于
2.1类模板没有自动类型推导的使用方式
如果想用类模板的类实例化对象需要用显示指定类型的方法
例如:
template <class T1, class T2>
class Person
{
public:
Person(T1 name, T2 age)
{
this->m_Name = name;
this->m_Age = age;
}
T1 m_Name;
T2 m_Age;
}
void Test1()
{
//Person p("孙悟空", 1000); 错误,无法用自动类型推导
Person<string, int> p("孙悟空", 1000); //正确 只能用显示指定类型
}
2.2类模板在模板参数列表中可以有默认参数
例如:template <class T1 = string, class T2>
类模板与普通类最大的区别在于分文件编写的时候,类模板通常将函数声明和实现写到同一个文件,并且文件后缀名通常用.hpp