C++模板初阶

C++模板初阶

大家在做OJ题的时候可能看到过这种使用STL的样子:

其实这个就是我们今天要介绍的模板这个语法在STL中的应用

一.泛型编程

在介绍泛型编程的概念之前,先给大家看一下这个例子

比方说:我今天要分别实现int和int类型,double和double类型,char和char类型的Swap函数

那么我可能需要利用函数重载写出一下三份代码

cpp 复制代码
void Swap(int& a, int& b)
{
	 int tmp = a;
	 a = b;
	 b = tmp;
}
void Swap(double& a, double& b)
{
	 double tmp = a;
	 a = b;
	 b = tmp;
}
void Swap(char& a, char& b)
{
	 char tmp = a;
	 a = b;
	 b = tmp;;
}

这样也是可以做到的

但是这样好不好呢?

二.函数模板

1.概念

2.实例

因此,我们就可以这样优化那份Swap函数了

cpp 复制代码
//typename是用来定义模板参数关键字,也可以使用class(切记:不能使用struct代替class)
//template<typename T>//T:type
template<class T>//T:type
void Swap(T& a, T& b)
{
	T tmp = a;
	a = b;
	b = tmp;
}
int main()
{
	int a = 0, b = 1;
	Swap(a, b);//模板的实例化
	double c = 2.1, d = 3.1;
	Swap(c, d);
	char e = 'a', f = 'b';
	Swap(e, f);
	return 0;
}

下面问题来了,这三次调用Swap函数调用的是同一个函数吗?

答案是:并不是,为什么呢?

这就涉及到下面的函数模板的原理了

3.原理


因此,如果我的这个函数调用让编译器无法推演出T的类型,也就是下面这种情况,编译器就会报错

(这里我使用Add这个函数来为下面做例子,关于为什么不用Swap函数我会在介绍的时候一并说明的)

注意:在模板中,编译器一般不会进行隐式类型转换操作,

因为一旦转化出问题,编译器就需要背黑锅

能不能通过某种手段来解决这种错误呢?

可以使用下面介绍的函数模板的实例化

4.函数模板的实例化

1.隐式实例化

我们可以这样去做:

这里我通过强制类型转换

分别将i和d这两个变量转换为int和double类型,成功完成了相加

但是:

那么为什么Swap函数就不可以呢?

我们之前在C++入门-引用中提到过:

那么接下来我们来看第二种方法:显式实例化

2.显式实例化

这里<int>里面的int就是告诉编译器,这个T的类型就是int,同理<double>也是如此

那么这种方法能不能解决Swap呢?

当然不行啦,因为我这个Swap的报错是权限放大导致的报错,不是T的类型无法推演出来所导致的报错

注意:

1.当我们无法通过传参的方式来让函数模板推演出T的类型的时候,才是显式实例化真正的用武之地:

2.模板参数列表也可以定义很多T

5.模板参数的匹配原则

1.一个非模板函数可以和一个同名的函数模板同时存在,而且该函数模板还可以被实例化为这个非模板函数

  1. 对于非模板函数和同名函数模板:

如果其他条件都相同,在调动时会优先调用非模板函数而不会从该模板产生出一个实例

如果模板可以产生一个具有更好匹配的函数 , 那么将选择模板

  1. 模板函数不允许自动类型转换,但普通函数可以进行自动类型转换

三.类模板

1.类模板的引出

我们在学习数据结构的时候经常会用到

cpp 复制代码
typedef int STDateType;

等等这样的typedef

但是typedef也不能解决所有问题:

比如这种问题:

如果我今天定义两个栈对象,一个存int,一个存double,这种情况下typedef就无法解决了,

那么该怎么办呢?

这个时候就要用到类模板了

2.实例

语法形式跟函数模板类似

不过:

也就是说这里的真正的类是

Stack<int> Stack<double> Stack<char> Stack<A>

以上就是C++模板初阶的全部内容,希望能对大家有所帮助!

相关推荐
老四啊laosi5 小时前
[C++进阶] 24. 哈希表封装unordered_map && unordered_set
c++·哈希表·封装·unordered_map·unordered_set
妙为6 小时前
银河麒麟V4下编译Qt5.12.12源码
c++·qt·国产化·osg3.6.5·osgearth3.2·银河麒麟v4
史迪仔01129 小时前
[QML] QML IMage图像处理
开发语言·前端·javascript·c++·qt
会编程的土豆10 小时前
【数据结构与算法】再次全面了解LCS底层
开发语言·数据结构·c++·算法
低频电磁之道10 小时前
解决 Windows C++ DLL 导出类不可见的编译错误
c++·windows
君义_noip12 小时前
信息学奥赛一本通 4150:【GESP2509七级】⾦币收集 | 洛谷 P14078 [GESP202509 七级] 金币收集
c++·算法·gesp·信息学奥赛·csp-s
Ricky_Theseus12 小时前
静态链接与动态链接
c++
澈20712 小时前
双指针,数组去重
c++·算法
小辉同志13 小时前
207. 课程表
c++·算法·力扣·图论
feng_you_ying_li13 小时前
C++11,{}的初始化情况与左右值及其引用
开发语言·数据结构·c++