【C++】模板初阶

一、函数模板

【函数模板格式】

template<typename T1, typename T2,......,typename Tn>

例子:

复制代码
template<typename T>
void Swap(T& left, T& right)
{
	T temp = left;
	left = right;
	right = temp;
}

int main()
{
	int a = 1, b = 2;
	double c = 1.1, d = 1.2;
	Swap(a,b);
	Swap(c, d);
	return 0;
}

注意:typename是用来定义模板参数关键字,也可以使用class

【函数模板的原理】

当调试的时候在模板层,但是底层依旧是调用Swap<int>,Swap<double>,不是Swap<T>

函数模板是一个蓝图,它本身并不是函数,是编译器用使用方式产生特定具体类型函数的模具。所以其实模板就是将本来应该我们做的重复的事情交给了编译器

在编译器编译阶段,对于模板函数的使用,**编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用。**比如:当用double类型使用函数模板时,编译器通过对实参类型的推演,将T确定为double类型,然后产生一份专门处理double类型的代码,对于字符类型也是如此。

1.1 函数模板的实例化

不同类型的参数使用函数模板 时,称为函数模板的实例化。

模板参数实例化分为:隐式实例化和显式实例化。

【隐式实例化】

让编译器根据实参推演模板参数的实际类型

复制代码
template<class T>
T Add(const T& left, const T& right)
{
	return left + right;
}

int main()
{
	int a1 = 10, a2 = 20;
	double d1 = 10.0, d2 = 20.0;
	Add(a1, a2);
	Add(d1, d2);
	return 0;
}

	//如果想使用模板,且传递的是不同类型的变量
	//1.强制类型转换
	//2.显示实例化
	cout << Add((double)a1,d1) << endl;
	cout << Add(a1,(int)d1) << endl;

【显示实例化】

在函数名后的<>中指定模板参数的实际类型

复制代码
	cout << Add<int>(a1, d1) << endl;
	cout << Add<double>(a2, d1) << endl;

还能这么玩

复制代码
template<typename T1,typename T2>
auto Add(const T1& left, const T2& right)
{
	return left + right;
}//用了auto让他推算类型
int main()
{
	int a1 = 10, a2 = 20;
	double d1 = 10.3, d2 = 20.4;

	// 显示实例化,不再让你去推演参数
	cout << Add<double,int>(a1, d1) << endl;
	cout << Add<int,int>(a1, d1) << endl;
	return 0;
}

无法推演出类型时,使用实例化,例如

复制代码
template<typename T>
T* func(int a)
{
	T* p = (T*)operator new(sizeof(T));
	new(p)T(a);
	return p;
}
int main()
{
	func<int>(1);
	return 0;
}

1.2 模板参数的匹配原则

复制代码
int Add(const int& left, const int& right)
{
	cout << "int Add(const int& left, const int& right)" << endl;
	return left + right;
}
template<typename T>
T Add(const T& left, const T& right)
{
	cout << "T Add(const T& left, const T& right)" << endl;
	return left + right;
}
template<typename T1,typename T2>
auto Add(const T1& left, const T2& right)
{
	cout << "auto Add(const T1& left, const T2& right)" << endl;
	return left + right;
}
//优先匹配  普通函数+参数匹配
//其次匹配  参数匹配+函数模板
//最后      类型转换一下也能用
int main()
{
	Add(1, 2);
	Add(1.1, 2.1);
	Add(1, 2.2);
	int a = 10;
	double b = 10.1;
	Add(a, (int)b);
	Add(a, b);
	return 0;
}

二、类模板

【类模板的定义格式】

复制代码
template<class T1, class T2, ..., class Tn>
class 类模板名
{
	// 类内成员定义
};

template<typename T>
class Stack
{
public:
	Stack(size_t capacity = 4)
	{
		_array = (T*)malloc(sizeof(T) * capacity);
		if (nullptr == _array)
		{
			perror("malloc申请空间失败");
			return;
		}
		_capacity = capacity;
		_size = 0;
	}

	void Push(const T& data);
private:
	T* _array;
	size_t _capacity;
	size_t _size;
};

声明和定义分离,我们如何定义呢?

复制代码
template<typename T>
void Stack<T>::Push(const T& data)
{
	_array[_size] = data;
	++_size;
}

模板不建议声明和定义分离到.h和.cpp会出现链接错误,具体后面会写

复制代码
int main()
{
	Stack<int> st1; //类的对象类型int
	Stack<double> st1; //类的对象类型double
	return 0;
}
相关推荐
2401_8920709821 小时前
【Linux C++ 日志系统实战】LogFile 日志文件管理核心:滚动策略、线程安全与方法全解析
linux·c++·日志系统·日志滚动
yuzhuanhei21 小时前
Visual Studio 配置C++opencv
c++·学习·visual studio
不爱吃炸鸡柳21 小时前
C++ STL list 超详细解析:从接口使用到模拟实现
开发语言·c++·list
十五年专注C++开发1 天前
RTTR: 一款MIT 协议开源的 C++ 运行时反射库
开发语言·c++·反射
‎ദ്ദിᵔ.˛.ᵔ₎1 天前
STL 栈 队列
开发语言·c++
2401_892070981 天前
【Linux C++ 日志系统实战】高性能文件写入 AppendFile 核心方法解析
linux·c++·日志系统·文件写对象
郭涤生1 天前
STL vector 扩容机制与自定义内存分配器设计分析
c++·算法
༾冬瓜大侠༿1 天前
vector
c语言·开发语言·数据结构·c++·算法
cccyi71 天前
【C++ 脚手架】etcd 的介绍与使用
c++·服务发现·etcd·服务注册
liu****1 天前
第16届省赛蓝桥杯大赛C/C++大学B组(京津冀)
开发语言·数据结构·c++·算法·蓝桥杯