lesson05-C++模板

个人主页: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 类名,有参数列表吗?显然没有,所以才需要我们显式去传模板参数。 

同时我们需要注意的是:

类模板并不是真正的类,就像模板函数一样,只有将他实例化后才是函数,才是真正的类。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Lei宝啊

觉得博主写的有用就鼓励一下吧

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值