目录
1. 非类型模板参数
2. 模板的特化
2.1 概念
2.2 函数模板特化
2.3 类模板特化
2.3.1 全特化
3 模板分离编译
3.1 什么是分离编译
3.2 模板的分离编译
4. 模板总结
有需要的老哥可以先看看模板的介绍:http://t.csdn.cn/2TkUYhttp://t.csdn.cn/2TkUY
模板参数分为 类型形参与 非类型形参。类型形参:出现在模板参数列表中,跟在 class 或者 typename 之类的参数类型名称。非类型形参:就是用一个常量作为类 ( 函数 ) 模板的一个参数,在类 ( 函数 ) 模板中可将该参数当成常量来使用。
namespace grm
{
// 定义一个模板类型的静态数组
template
class array
{
public:
T& operator[](size_t index) { return _array[index]; }
const T& operator[](size_t index)const { return _array[index]; }
size_t size()const { return _size; }
bool empty()const { return 0 == _size; }
private:
T _array[N];
size_t _size;
};
}
int main()
{
grm::array arr;
return 0;
}
就比如上面这种,模板参数多了一个int类型的N,实例化出arr后,为其开辟的就是100个空间大小。
注意:1. 浮点数、类对象以及字符串是不允许作为非类型模板参数的。2. 非类型的模板参数必须在编译期就能确认结果。
像下面这种写法编译器会直接报错的:
函数模板的特化步骤:1. 必须要先有一个基础的函数模板2. 关键字template后面接一对空的尖括号<>3. 函数名后跟一对尖括号,尖括号中指定需要特化的类型4. 函数形参表: 必须要和模板函数的基础参数类型完全相同,如果不同编译器可能会报一些奇怪的错误。
// 函数模板 -- 参数匹配
template
bool Less(T left, T right)
{
return left < right;
}
// 对Less函数模板进行特化
template<>
bool Less(Date* left, Date* 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;
}
但是我个人感觉使用这种方法还不如直接重载一个版本。
bool Less(Date* left, Date* right)
{
return *left < *right;
}
全特化即是将模板参数列表中所有的参数都确定化。
template
class Data
{
public:
Data() {cout<<"Data" <
class Data
{
public:
Data() {cout<<"Data" < d1;
Data d2;
}
我们可以调试起来看看:发现d1会调用第一种模板,而d2会调用第二种模板参数:
像我们之前实现日期类的比较就可以用这种版本:
2.3.2 偏特化
偏特化:任何针对模版参数进一步进行条件限制设计的特化版本。比如对于以下模板类:
template
class Data
{
public:
Data() {cout<<"Data" <
// 将第二个参数特化为int
template
class Data
{
public:
Data() {cout<<"Data" <
//两个参数偏特化为指针类型
template
class Data
{
public:
Data() {cout<<"Data" <
class Data
{
public:
Data(const T1& d1, const T2& d2)
: _d1(d1)
, _d2(d2)
{
cout<<"Data" < d1; // 调用特化的int版本
Data d2; // 调用基础的模板
Data d3; // 调用特化的指针版本
Data d4(1, 2); // 调用特化的指针版本
}
一个程序(项目)由若干个源文件共同实现,而每个源文件单独编译生成目标文件,最后将所有目标文件链接起来形成单一的可执行文件的过程称为分离编译模式。
假如有以下场景,模板的声明与定义分离开,在头文件中进行声明,源文件中完成定义:
// a.h
template
T Add(const T& left, const T& right);
// a.cpp
template
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;
}
我们来运行一下:
发现报了链接错误,为什么呢?
我们可以简单来分析一下:程序处理阶段分为4个过程:
预处理 编译 汇编 链接
想了解这四个步骤的具体详解可以参照博主的这一篇文章:
http://t.csdn.cn/QA43Thttp://t.csdn.cn/QA43T在a.cpp中,编译器没有看见对Add模板函数的实例化(原因C++标准明确表示,当一个模板不被用到的时侯它就不该被实例化出来),因此是不会生成具体的函数,当链接时找函数地址时就会因找不到而报错,具体处理方式有两种:
1 模板定义的位置显式实例化:
template<>
int Add(const int& left, const int& right)
{
return left + right;
}
template<>
double Add(const double& left, const double& right)
{
return left + right;
}
但是这种方式就没有了范型编程的优势了,所以这种方式并不可取。
2 将声明和定义放到一个文件 "xxx.hpp" 里面或者xxx.h其实也是可以的。(推荐使用放在xxx.h中)
//a.h
template
T Add(const T& left, const T& right)
{
return left + right;
}
在分离式编译的环境下,编译器编译某一个.cpp文件时并不知道另一个.cpp文件的存在,也不会去查找(当遇到未决符号时它会寄希望于连接器)。这种模式在没有模板的情况下运行良好,但遇到模板时就傻眼了,因为模板仅在需要的时候才会实例化出来,所以,当编译器只看到模板的声明时,它不能实例化该模板,只能创建一个具有外部连接的符号并期待连接器能够将符号的地址决议出来。然而当实现该模板的.cpp文件中没有用到模板的实例时,编译器懒得去实例化,所以,整个工程的.obj中就找不到一行模板实例的二进制代码,于是连接器也黔驴技穷了。
【优点】
1. 模板复用了代码,节省资源,更快的迭代开发, C++ 的标准模板库 (STL) 因此而产生2. 增强了代码的灵活性【缺陷】1. 模板会导致代码膨胀问题,也会导致编译时间变长2. 出现模板编译错误时,错误信息非常凌乱,不易定位错误