【C++进阶之路】模板

news2025/1/8 17:09:52

前言

假如需要你写一个交换函数,交换两个相同类型的值,这时如果交换的是int 类型的值,你可能会写一个Swap函数,其中参数是两个int类型的,假如再让你写一个double类型的呢?你可能又要写一个Swap的函数重载,参数是两个类型double的,假如再让你写一个char类型的呢?你可能……这样会有尽头吗?没有,因为还有自定义类型呢!那我们如何解决这样的问题呢?——模板(主要功能是实现通用)

  • 补充:泛型编程:编写与类型无关通用代码,是代码复用的一种手段。模板是泛型编程的基础。

一. 函数模板

基本用法

①定义模板参数类型名

template<typename T1, typename T2,......,typename Tn>
//这里的typename也可以换为class
template<class T1, class T2,......,class Tn>
  • 切记: 不能换为struct

②函数模板的实现

  • 比如我们实现一个交换函数的模板
template<typename T>
void Swap(T& n1, T& n2)
{
	T tmp = n1;
	n1 = n2;
	n2 = tmp;
}

用如下几个例子实验一下:
例1:

class Date
{
public:

	void Print()
	{
		cout << _year << "年" << _month << "月" << _day << "日" << endl;
	}
	Date(int year,int month,int day)
		:_year(year)
		,_month(month)
		,_day(day)
	{

	}
private:
	int _year;
	int _month;
	int _day;
};
int main()
{
	int x = 0,y = 1;
	double x1 = 1.1, y1 = 2.2;
	Date x2(1949, 10, 1), y2(2023, 5, 23);
	Swap(x, y);
	Swap(x1, y1);
	Swap(x2, y2);
	x2.Print();
	y2.Print();
	cout << "x:" << x << " y:" << y << endl;
	cout << "x1:" << x1 << " y1:" << y1 << endl;
	return 0;
}

运行结果:
在这里插入图片描述

  • 可以看到这是交换了。

到这里我们又有一个疑问——调用的是一个函数吗?
继续分析——转到反汇编
在这里插入图片描述

  • 可以看到——调用的Swap的地址是不一样的!
  • 因此:调用的不是一个函数。
  • 如何解释呢?

换汤不换药,换的是类型,不变的交换的思想。
图解:
在这里插入图片描述

其实我们也跟根本不用自己实现交换函数,直接用库里的就行了。在这里插入图片描述

  • 不过库里的是全小写,我们平常实现的是首字母大小。

③模板的实例化


隐式实例化

  • 编译器根据实参推演模板参数的实际类型
template <typename T>
T Func(const T& n1, const T& n2)
{
	return n1 + n2;
}
int main()
{
	int x = 0,y = 1;
	Func(x, y);
	return 0;
}
  • 就好比这里传参并没有明确的说明x的类型,但是编译器会自动推导实参的类型。

显示实例化

  • 在函数名后的<>中指定模板参数的实际类型
template <typename T>
T Func(const T& n1, const T& n2)
{
	return n1 + n2;
}

int main()
{
	int x = 0,double y = 1;
	Func<int>(x, y);
	//这里会将y强制类型转换为一个double类型的常量,因此函数参数的const不可省去。
	return 0;
}
  • 像这样如果,编译器推导不出来类型,这时就需要我们显示声明一下,其中的类型是什么,否则编译器会直接报错。
  • 那除了显示示例化还有什么办法么?——强制类型转换。
template <typename T>
T Func(const T& n1, const T& n2)
{
	return n1 + n2;
}

int main()
{
	int x = 0,double y = 1;
	Func(x, (int)y);
	//这里会将y强制类型转换为一个double类型的常量,因此函数参数的const不可省去。
	return 0;
}
  • 强转的结果必然是使两个参数类型保持一致,如果强转后的参数类型不一致,那么编译器还是会报错。

还有一种情况是我们必须显示示例化的——模板参数做返回值
示例:

template <typename T>
T Func(int x, int y)
{
	return x + y;
}

int main()
{
	int x = 0;
	int y = 1;
	Func<int>(x,y);
	return 0;
}
  • 这就好比你要使用函数模板,得先让编译器知道或者能推导出模板参数的类型,那么返回值写上模板参数的话,编译器是无从下手的,那么只有显式实例化了,编译器才知道返回类型,才会给你生成指定类型的函数。

④函数模板与普通函数的调用规则

1.能用普通函数就用普通函数,用不了再用函数模板。

int add(int x, int y)
{
	return x + y;
}

template <typename T1,typename T2>
int add(T1 x, T2 y)
{
	return x + y;
}

int main()
{
	add(1, 2);//这里调用的非模板
	add(1.0, 1);//这里使用的是模板生成的函数
	return 0;
}

看反汇编:
在这里插入图片描述

2.非模板函数与模板部分重复都可以使用。

int add(int x, int y)
{
	return x + y;
}
template <typename T1,typename T2>
int add(T1 x, T2 y)
{
	return x + y;
}
int main()
{
	add(1, 2);//这里是调用add函数,因为能用普通函数就用普通函数
	add<int>(1, 1);这里是用的add模板函数实例化生成的函数。
	//这样也可以
	add<>(1, 1);//这算是空模板,只是为了调用模板而已。
	return 0;
}

3.普通函数支持类型转换,而模板要严格按照类型。

template<typename T>
int myAdd(T a, T b)
{
	cout << "template function" << endl;
	return a + b;
}

int myAdd(char a, char b)
{
	cout << "normal function" << endl;
	return a + b;
}
void test()
{
	int a = 10;
	int b = 20;
	char c1 = 'a';
	char c2 = 'b';

	myAdd(a, c1);//这里调用的是函数,这里的a会自动转换为char类型的
	myAdd(a, b);//两个都是int类型的调用模板函数
	myAdd(c1, b);//这里会发生类型转换,b的int类型会转换为char类型的
	myAdd(c1, c2);
	myAdd<>(c1, c2);//这里是显示实例化,自然会调用函数模板
}
int main()
{
	test();
	return 0;
}

  • 总结:

1.普通函数优先调用
2.普通函数支持类型转换
3.模板必须严格的类型匹配
4.如果函数模板可以产生一个更好的匹配,那么选择模板。
5.可以用空模板或者模板实例化让编译器只调用模板

二. 类模板

  • 简而言之就是类是通用的,比如一个栈既要存int类型的,又要存double类型等等。
  • 说明:类模板不是具体的类,是用来生成类对象的模具。

  • 补充:template定义的模板参数只在最近的一个大括号内有效。
#include<iostream>
using namespace::std;
template <typename T>
class Stack
{
public:
	Stack(int capacity = 4)
	{
		cout << "Stack()" << endl;
		T* tmp = new T[capacity];
		_arr = tmp;
		_top = 0;
		_capacity = capacity;
	}
	void PushBack(T x);

private:
	T* _arr;
	int _top;
	int _capacity;
};
//这里放在类外进行定义。
template <typename T>
//这里要说明的是必须写Stack<T>这是具体的类,而不能只写Stack。
void Stack<T>::PushBack(T x)
{
	if (_top == _capacity)
	{
		T* tmp = (T*)realloc(_arr, sizeof(T) * _capacity * 2);
		if (tmp == NULL)
		{
			perror("realloc fail");
			exit(-1);
		}
		else
		{
			_arr = tmp;
			_capacity *= 2;
		}
	}
	_arr[_top++] = x;
}
int main()
{
	Stack<int> stack1;//这要说明使用的数据类型,也就是类模板示例化。
	return 0;
}

// 注意:类模板
//1.类名——Stack
//2.类型 ——Stack<T>
  • 类模板实例化得到的类叫模板类

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/558953.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

运营-14.优惠券规则

优惠券使用场景 1. 需要先领取&#xff1b; 2. 在购买商品的时候可以抵用部分费用&#xff1b; 3. 在有效期内可以随时使用&#xff1b; &#xff08;根据场景和类型会有区别&#xff09; 优惠券的优势 1. 可以控制数量&#xff0c;确保活动成本可控&#xff1b; 2. 使用灵活&a…

「超强」ChatGPT撰写的艾思科技软件定制开发行业可行性报告分析

I. 引言 - 报告目的和范围 本报告的目的是分析山东艾思软件科技有限公司在定制软件开发行业的可行性&#xff0c;并提供相关建议和指导。本报告主要分析定制软件开发行业的市场概况、市场规模和增长潜力、市场需求、供应情况以及市场财务可行性。本报告旨在为山东艾思软件科技…

BIM建模|什么样的计算机可以支撑BIM应用?

BIM建模&#xff5c;什么样的计算机可以支撑BIM应用&#xff1f; 随着国内BIM技术的不断发展&#xff0c;越来越多企业和个人开始重视BIM&#xff0c;而作为BIM应用的数据生产载体&#xff0c;计算机硬件实力显得尤为重要。 现阶段BIM软件种类繁多&#xff0c;不同项目、不同…

Mysql安装与卸载(Windows版本)

Mysql的安装 这里使用的Mysql版本是8.0.26 界面操作描述信息 1. 接受条款&#xff0c;下一步&#xff0c;准备开始安装image-202111220927156542. 选择Custom&#xff0c;自定义安装&#xff0c;Nextimage-202111221638172083. 以自己的操作系统为准&#xff0c;不过大多数都…

批发零售商城小程序开发功能优势有哪些?

阿里发展到今天可能是很多人都意想不到的&#xff0c;谁能行到当初马云的一个大胆决定会让其成为批发零售行业的龙头呢。随着互联网技术的深入发展&#xff0c;现在越来越的商家企业也都寻求新的经营发展方式&#xff0c;批发零售商城小程序开发作为一种新型的电商模式&#xf…

达梦数据库安装教程

目录 安装教程 安装前准备 新建dmdba用户 修改文件打开最大数 挂载镜像 新建安装目录 数据库安装 配置环境变量 配置实例 注册服务 启动停止服务 启动 查看端口 停止 数据库目录结构介绍 数据库安装目录 达梦数据库 DM8下载地址产品下载 | 达梦数据库 (dameng.…

【Vue基础】Vue路由,实现页面跳转

一、需求说明 点击不同的模块实现页面跳转&#xff0c;如下点击“员工管理”右侧会显示员工管理页面&#xff0c;如下图1&#xff1b;点击“入住信息”右侧会显示入住信息&#xff0c;如下图二 二、涉及文件 1、 主要上图在这几个文件中修改相关代码 2、知识点整理 1&#x…

25K 入职腾讯的那天,我特么哭了

悲催的经历&#xff1a; 先说一下自己的个人情况&#xff0c;计算机专业&#xff0c;17 年本科毕业&#xff0c;一毕业就进入了“阿里”软件测试 岗(进去才知道是接了个阿里外包项目&#xff0c;可是刚毕业谁知道什么外包不外包的)。 更悲催的是&#xff1a;刚入职因为家里出…

工厂蓝牙定位技术的原理、应用场景、优势及潜在问题

蓝牙定位技术是近年来在工业领域中得到广泛应用的一项技术。随着工业自动化的快速发展和物联网技术的普及&#xff0c;工厂蓝牙定位成为了提高生产效率、优化生产流程和管理的重要工具。本文将详细介绍工厂蓝牙定位技术的原理、应用场景以及其在工业生产中的优势。 首先&#x…

redis集群读写,容错切换,从属调整,扩容,缩容

rediscluster 读写一定要注意redis实例的区间实例范围。需要路由到位。 比如 hashsolthash(k1) mod 1638412706,而12706槽位不在6391上&#xff0c;在6393上。 如何让rediscluster 路由到槽呢&#xff1f; redis-cli命令尾部加上 -c即可。防止路由失效。如果k1不在6391上&am…

企业如何将采购成本最小化?

从人员成本到运输和手续费&#xff0c;采购成本涵盖了广泛的费用&#xff0c;这些费用可能会迅速增加。这就是为什么要有一个明确的采购流程&#xff0c;鼓励竞争性招标&#xff0c;并使供应商轻松与你合作。但是&#xff0c;降低采购成本的最有效方法也许是通过实施一个采购软…

电力导线镭射光防外破预警系统

电力导线镭射光防外破预警系统 一、产品描述&#xff1a; 我司研发生产了一款型号为TLKS-PMG-WPI的电力导线镭射光防外破预警系统&#xff0c;是与电力设计时代发展升级的产品&#xff0c;在配电网需要实时在线监测的前景下&#xff0c;设备的稳定可靠性也是一种前所未有的挑战…

Linux 设备驱动程序(三)

系列文章目录 Linux 内核设计与实现 深入理解 Linux 内核&#xff08;一&#xff09; 深入理解 Linux 内核&#xff08;二&#xff09; Linux 设备驱动程序&#xff08;一&#xff09; Linux 设备驱动程序&#xff08;二&#xff09; Linux 设备驱动程序&#xff08;三&#xf…

WIN提权 令牌窃取进程注入

令牌窃取&#xff08;鸡肋玩意 2008包括2008以下&#xff09; 令牌&#xff0c;又叫token&#xff0c;是系统临时产生的秘钥&#xff0c;相当于账号密码&#xff0c;用来决定是否允许此次请求和判断此次请求是属于哪一个用户。 win7一下的版本可以尝试 这里使用msf上自带的令…

CCF-CSP 202006-1 线性分类器

根据高中知识我们知道&#xff0c;通过将点带入直线方程得到大于0或等于0或小于0&#xff0c;其中大于0与小于0的情况即对应两个点在直线的两边 据此&#xff0c;我们取最后一个输入的A点作为标准、最后一个输入的B点作为标准&#xff0c;记录下他们的>0还是<0&#xff0…

市面上常见的语音芯片的IO口有哪些作用

语音芯片的IO口有哪些作用&#xff1f; 语音芯片的IO口一般有多种用途&#xff0c;包括以下几种&#xff1a; 1. 语音输入&#xff1a;可以通过外部麦克风接口&#xff0c;将外部声音信号输入到语音芯片中&#xff0c;进行语音信号处理。 2. 语音输出&#xff1a;语音芯片可…

UTP网络编程入门案例

说明&#xff1a;UTP是面向无连接的&#xff0c;不可靠的协议&#xff0c;即传输数据时不会确定对方是否在线&#xff0c;优点是效率高。 DatagramSocket & DatagramPacket DatagramSocket是使用UDP协议的Socket&#xff0c;它的作用是接收和发送数据包&#xff1b; Dat…

数字信号处理6

昨天简单的复习了一下离散时间信号是什么以及系统的概念、系统要做的工作和系统中几个简单的原件&#xff0c;今天跟着昨天的内容继续学习。 一、离散时间系统的分类&#xff1a; 为什么要对系统进行分类呢&#xff0c;这就像是我们对函数进行分类一样&#xff0c;有些函数有…

提供免费样机模板素材的好网站推荐

说到原型样机模板&#xff0c;设计师当然并不陌生&#xff0c;因为经常在设计师完成作品后&#xff0c;为了更好地展示作品&#xff0c;通常将设计作品应用于真正的原型样机模板&#xff0c;可以快速在现实场景中展示设计作品&#xff0c;选择好的原型样机模板&#xff0c;操作…

超低功耗SUB 1G无线收发芯片+32位 Soc芯片UM2080F32 无线集抄/测温/养老院人员定位

UM2080F32是一款超低功耗 32 位 IoTP&#xff1a;ARMCortex-M0&#xff0c;64KB Flash&#xff0c;16KB SRAM&#xff0c;Sub-1GHz 射频收发器。工作于200MHz~960MHz 范围内。 UM2080F32 内部还集成了 CAN、12 位 SAR ADC、UART、SPI、QSPI、I2C 等通用外围通讯接口&#xff0c…