目录
1.非类型模板参数
模板参数分为类型形参与非类型形参。
类型形参即:出现在模板参数列表中,跟在class或者typename之类的参数类型名称。
非类型形参,就是用一个常量作为类(函数)模板的一个参数,在类(函数)模板中可将该参数当成常量来使用。注意: 这个非类型形参在模板内是不能被修改的。
这里是一个简单的示例,非类型形参在类内是可以使用的。但一旦对其进行修改,就会报错。
template<class T,size_t N=10>
class A
{
public:
A(T a=N)
:_a(a)
{}
void Print()
{
//N++;
//cout << N << endl;
cout << _a << endl;
}
private:
T _a;
};
int main()
{
A<int> a;
A<double> b(1.1);
a.Print();
b.Print();
return 0;
}
1. 浮点数、类对象以及字符串是不允许作为非类型模板参数的。
2. 非类型的模板参数必须在编译期就能确认结果。
2. 模板的特化
2.1 概念
通常情况下,使用模板可以实现一些与类型无关的代码,但对于一些特殊类型的可能会得到一些错误的结果,需要特殊处理,比如:实现了一个专门用来进行小于比较的函数模板
// 函数模板 -- 参数匹配
template<class T>
bool Less(T left, T right)
{
return left < right;
}
int main()
{
cout << Less(1, 2) << endl; // 可以比较,结果正确
Date d1(2022, 7, 7);
Date d2(2022, 7, 8);
cout << Less(d1, d2) << endl; // 可以比较,结果正确
Date* p1 = &d1;
Date* p2 = &d2;
cout << Less(p1, p2) << endl; // 可以比较,结果错误
return 0;
当我们传入Date*指针时,并不会比较指针指向的内容,而是比较两个指针的大小,针对这种情况,我们就需要做一些特殊化处理,这就是模板的特化。
模板特化分为函数模板特化与类模板特化 。
2.2 函数模板特化
函数模板的特化步骤:
1. 必须要先有一个基础的函数模板
2. 关键字template后面接一对空的尖括号<>
3. 函数名后跟一对尖括号,尖括号中指定需要特化的类型
4. 函数形参表: 必须要和模板函数的基础参数类型完全相同,如果不同编译器可能会报一些奇怪的错误。
对于2.1里的示例,我们将less函数进行模板特化处理。
有没有发现,函数模板的特化类似于函数重载的功能,但写法却比重载复杂。因此我们更加推荐直接将需要特殊化处理的函数直接给出,而不是使用函数模板特化。
2.3 类模板特化
2.3.1 全特化
全特化:即将模板参数列表里的参数全部都确定化。
下图就是全特化的示例。
2.3.2 偏特化
偏特化:任何针对模版参数进一步进行条件限制设计的特化版本
偏特化有以下两种表现方式:
部分特化
将模板参数类表中的一部分参数特化
参数更进一步的限制
偏特化并不仅仅是指特化部分参数,而是针对模板参数更进一步的条件限制所设计出来的一个特化版本
部分特化示例
template<int, class T2>
class A
{
public:
A(int a, T2 b)
:_a(a)
, _b(b)
{
cout << "template<int,class T2>" << endl;
}
void Print()
{
cout << _a << endl;
cout << _b << endl;
}
private:
int _a;
T2 _b;
};
参数更进一步限制的示例
限制传参为T1*,T2*类型
template<class T1, class T2>
class A<T1*,T2*>
{
public:
A(T1 a, T2 b)
:_a(a)
, _b(b)
{
cout << "template<class T1,class T2>" << endl;
}
void Print()
{
cout << _a << endl;
cout << _b << endl;
}
private:
T1 _a;
T2 _b;
};
2.3.3 类模板特化应用
#include<vector>
#include <algorithm>
template<class T>
struct Less
{
bool operator()(const T& x, const T& y) const
{
return x < y;
}
};
int main()
{
Date d1(2022, 7, 7);
Date d2(2022, 7, 6);
Date d3(2022, 7, 8);
vector<Date> v1;
v1.push_back(d1);
v1.push_back(d2);
v1.push_back(d3);
// 可以直接排序,结果是日期升序
sort(v1.begin(), v1.end(), Less<Date>());
vector<Date*> v2;
v2.push_back(&d1);
v2.push_back(&d2);
v2.push_back(&d3);
// 可以直接排序,结果错误日期还不是升序,而v2中放的地址是升序
// 此处需要在排序过程中,让sort比较v2中存放地址指向的日期对象
// 但是走Less模板,sort在排序时实际比较的是v2中指针的地址,因此无法达到预期
sort(v2.begin(), v2.end(), Less<Date*>());
return 0;
}
这里我们对less函数做特化处理,让less能够比较指针指向的内容大小。特化后的函数在2.2中有。
3. 模板分离编译
3.1 什么是分离编译
一个程序(项目)由若干个源文件共同实现,而每个源文件单独编译生成目标文件,最后将所有目标文件链接起来形成单一的可执行文件的过程称为分离编译模式。
3.2 模板的分离编译
假设有以下场景,模板的声明与定义分别在.h 与.cpp中。
// a.h
template<class T>
T Add(const T& left, const T& right);
// a.cpp
template<class T>
T Add(const T& left, const T& right)
{
return left + right;
}
// main.cpp
#include"a.h"
int main()
{
Add(1, 2);
Add(1.0, 2.0);
return 0;
}
3.3 解决方法
1. 将声明与定义放在同一文件。(推荐)
2. 模板定义显式实例化。(不推荐)
如果将模板显式实例化,不就失去了模板的作用吗。
4. 模板总结
【优点】
1. 模板复用了代码,节省资源,更快的迭代开发,C++的标准模板库(STL)因此而产生
2. 增强了代码的灵活性
【缺陷】
1. 模板会导致代码膨胀问题,也会导致编译时间变长
2. 出现模板编译错误时,错误信息非常凌乱,不易定位错误