【C++】模板初阶

news2025/1/12 18:10:04

文章目录

  • 一、泛型编程
  • 二、函数模板
    • 1、概念与格式
    • 2、底层原理
    • 3、实例化
    • 4、参数的匹配规则
  • 三、类模板
    • 1、概念与格式
    • 2、实例化

一、泛型编程

我们通过实现一个通用的交换函数来引入泛型编程:

void Swap(int& left, int& right)
{
	int temp = left;
	left = right;
	right = temp;
}

void Swap(double& left, double& right)
{
	double temp = left;
	left = right;
	right = temp;
}

void Swap(char& left, char& right)
{
	char temp = left;
	left = right;
	right = temp;
}

在C语言阶段,要实现一个通用的交换函数我们只能通过定义对应不同参数类型的多个函数来实现,而且各函数的函数名不能相同,比如 Swapi、Swapd、Swapc;到了C++阶段,我们可以通过函数重载来定义多个参数类型不同但函数名相同的函数来实现,但是函数重载有以下几个缺陷:

  • 重载的函数仅仅是参数类型不同,代码复用率比较低,并且只要有新类型出现时,就需要用户自己增加对应的函数;
  • 代码的可维护性比较低,一个出错可能所有的重载均出错;

那么我们能否给编译器提供一个模型,让编译器能够根据不同的参数类型自动利用该模型来生成对应函数呢?就像浇筑一样,我们可以根据同一个浇筑模具来浇筑出不同类型的模具;image-20221019105245360

如果在C++中,也能够存在这样一个模具,通过给这个模具中填充不同材料 (类型),来获得不同材料的铸件 (即生成具体类型的代码),那将会节省许多头发。好消息是C++中确实存在类似的东西。

**泛型编程:编写与类型无关的通用代码,是代码复用的一种手段。模板是泛型编程的基础。**模板一共分为两类 – 函数模板和类模板。image-20221019151054330


二、函数模板

1、概念与格式

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

模板函数的定义格式如下:

template <class 类型参数1, class 类型参数2, ...>
返回值类型  模板名(形参表)
{
    函数体
}

其中的 class 关键字也可以用 typename 关键字来替换,如下:

template <typename 类型参数1, typename 类型参数2, ...>
返回值类型  模板名(形参表)
{
    函数体
}

上面的 template 为模板关键字,class/typename 关键字用于指定模板参数中的类型,类型参数用于代表该类型 (我们一般使用 T 作为类型参数)

我们以Swap函数为例:

template <class T>
void Swap(T& left, T& right)
{
	T tmp = left;
	left = right;
	right = tmp;
}

有了函数模板,我们就可以使用不同类型的参数来调用同一函数了:image-20221019153909465

2、底层原理

函数模板是一个蓝图,它本身并不是函数,所以当我们实际调用时编译器会根据传入的实参类型来推演生成对应类型的函数以供调用,此过程在编译阶段完成;比如当用double类型使用函数模板时,编译器通过对实参类型的推演,将T确定为double类型,然后产生一份专门处理double类型的代码,对于字符类型和整形也是如此。image-20221019175748876

image-20221019180128101

3、实例化

用不同类型的参数使用函数模板时,称为函数模板的实例化。模板参数实例化分为:隐式实例化和显式实例化

1、隐式实例化:让编译器根据实参推演模板参数的实际类型

image-20221019180628636

如上图,我们调用 Add 函数模板时并不需要显式指定 T 为 int 或 double 类型,编译器会根据实参类型自动去推演模板参数的类型,然后实例化出对应函数。

注意:我们在使用函数模板时需要避免下面这种情况:image-20221019182429246

上述语句不能通过编译,因为在编译期间,当编译器看到该实例化时,需要推演其实参类型通过实参a1将T推演为int,通过实参d1将T推演为double类型,但模板参数列表中只有一个T,编译器会无法确定此处到底该将T确定为int 或者 double类型而报错。

注意:在模板中,编译器一般不会进行类型转换操作,因为一旦转化出问题,编译器就需要背黑锅。

对于上面这种情况一共有三种解决方式:用户对实参进行强转、增加模板参数或者显示实例化;

a. 用户对实参进行强转:image-20221019182526468

需要注意的是函数的形参必须用 const 修饰,因为 (int)d1 传递给形参的是一个临时变量,而临时变量具有常性,需要用 const 引用来接收。

b. 增加模板参数:image-20221019182632254

但是其实这里又会产生一个新的问题 – 函数返回值的类型到底应该用T1还是T2。

c. 显示实例化。

2、显示实例化:在函数名后的<>中指定模板参数的实际类型

image-20221019182813137

显式示例化的原理和用户强转类似,只不过这里是编译器自动将 d1 强转为 int 然后传递给形参,或者将 a1 强转为 double 传递给形参;同时,这里函数的形参也必须用 const 修饰。

4、参数的匹配规则

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

// 专门处理int的加法函数
int Add(int left, int right)
{
	cout << "normal add" << endl;
	return left + right;
}

// 通用加法函数
template<class T>
T Add(T left, T right)
{
	cout << "template add" << endl;
	return left + right;
}

image-20221020100404112

可以看到,当一个非模板函数可以和一个同名的函数模板同时存在时,如果我们不显式实例化,编译器会去调用非模板函数,而不会去实例化模板;如果我们显示实例化,编译器会调用通过函数模板实例化得到的函数。

另外,显示实例化后程序能够正常运行,也侧面的说明了通过函数模板实例化出的函数与非模板函数 (普通函数) 的函数名修饰规则不同,否则会发生编译错误。

2、对于非模板函数和同名函数模板,如果其他条件都相同,在调动时会优先调用非模板函数而不会从该模板产生出一个实例;如果模板可以产生一个具有更好匹配的函数, 那么将选择模板。

// 专门处理int的加法函数
int Add(int left, int right)
{
	cout << "normal Add" << endl;
	return left + right;
}

// 通用加法函数
template<class T1, class T2>
T1 Add(T1 left, T2 right)
{
	cout << "template Add" << endl;
	return left + right;
}

image-20221020101645583

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


三、类模板

1、概念与格式

和非模板函数一样,我们以前实现的类也有一个缺点 – 同一个类只能示例一种类型的对象,我们以 Stack 为例:

typedef int STDateType;
class Stack
{
public:
	Stack(int capacity = 4)
		:_top(0)
		, _capacity(capacity)
	{
		_a = (STDateType*)malloc(sizeof(STDateType) * capacity);
		if (_a == nullptr)
		{
			perror("malloc fail\n");
			exit(-1);
		}
	}

	~Stack()
	{
		free(_a);
		_a = NULL;
		_top = _capacity = 0;
	}

	void Push(STDateType x)
	{
		_a[_top++] = x;
	}

private:
	STDateType* _a;
	int _top;
	int _capacity;
};

image-20221020102950701

image-20221020103125677

可以看到,虽然我们可以通过修改 STDateType 来让 _a 存储不同类型的数据,但是在同一个项目中 STDateType 只能被指定为一种类型;所以不能处理上面这种情况;那么和模板函数一样,我们也需要一个类模板,使得我们能够根据同一个模板来实例化出不同的类。

类模板的定义格式如下:

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

有了类模板,我们就可以将 Stack 修改为下面这样:

// 注意:Stack不是具体的类,是编译器根据被实例化的类型生成具体类的模具
template <class T>
class Stack
{
public:
	Stack(int capacity = 4)
		:_top(0)
		, _capacity(capacity)
	{
		_a = (T*)malloc(sizeof(T) * capacity);
		if (_a == nullptr)
		{
			perror("malloc fail\n");
			exit(-1);
		}
	}

	~Stack()
	{
		free(_a);
		_a = NULL;
		_top = _capacity = 0;
	}

	void Push(T x)
	{
		_a[_top++] = x;
	}

private:
	T* _a;
	int _top;
	int _capacity;
};

2、实例化

类模板实例化与函数模板实例化不同,类模板实例化需要在类模板名字后跟<>,然后将实例化的类型放在<>中即可,即类模板必须显示实例化;

这里有一个特别容易出错的地方:类模板名字不是真正的类,而实例化的结果才是真正的类;在之前我们说过,C++中类名就是类型,但是类模板和普通的类不同 – 类模板不是具体的类,是编译器根据被实例化的类型生成具体类的模具;即只有我们对类模板进行显示实例化之后编译器才会生成具体的类,而这个类才是我们能够正常使用的类;所以实例化的结果才是真正的类。

Stack -- 类名;
Stack<int> -- 类型;
Stack<double> -- 类型;

image-20221020105304978

image-20221020105339869


ps:模板还有一些其他的内容,比如非类型模板参数、类模板的特化、模板的分离编译等,我们在学完 STL 初阶后再进行学习。


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

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

相关文章

Linux 命令(147) —— truncate 命令

文章目录1.命令简介2.命令格式3.选项说明4.常用示例参考文献1.命令简介 truncate 将文件的大小缩小或扩展到指定的大小。 如果指定的文件不存在将被创建。 如果文件大于指定的大小&#xff0c;则会丢失额外的数据。如果较短&#xff0c;它将被扩展&#xff0c;扩展的稀疏部分…

【牛客刷题--SQL篇】多表查询组合查询SQL25 查找山东大学或者性别为男生的信息

&#x1f496;个人主页&#xff1a;与自己作战 &#x1f4af;作者简介&#xff1a;CSDN博客专家、CSDN大数据领域优质创作者、CSDN内容合伙人、阿里云专家博主 &#x1f49e;牛客刷题系列篇&#xff1a;【SQL篇】】【Python篇】【Java篇】 &#x1f4cc;推荐刷题网站注册地址&a…

Python数据分析与挖掘————图像的处理

系列文章目录 文章目录系列文章目录前言图片的马赛克一.安装matplotlib&#xff0c;numpy等模块二.马赛克图片一.导入图片二.定位区域三.图片的合成图片拼接图像的灰度化一.max()方法二.min&#xff08;&#xff09;方法三.平均值法mean()函数四.加权平均值法图片的分割总结源代…

基于tauri+vue3.x多开窗口|Tauri创建多窗体实践

最近一种在捣鼓 Tauri 集成 Vue3 技术开发桌面端应用实践&#xff0c;tauri 实现创建多窗口&#xff0c;窗口之间通讯功能。 开始正文之前&#xff0c;先来了解下 tauri 结合 vue3.js 快速创建项目。 tauri 在 github 上star高达53K&#xff0c;而且呈快速增长趋势。相比elect…

DDoS报告团伙规模

攻击资源活跃度分析 在攻击源活时间的监测中发现&#xff0c;和 2019 年趋势一致&#xff0c;存活时间大于 10 天的攻击资源占比 11%。像这种能够长期被控制的肉鸡大部分都是物联网 设备&#xff0c;物联网设备大都存在设备系统老&#xff0c;人员维 护少&#xff0c;更新慢等…

vue当中的事件处理

1.绑定监听v-on 最简单的一个绑定监听的事件 <body><div id"root"><h1>my name is {{name}}</h1><button v-on:click"showInfo">click me</button></div><script type"text/javascript">Vue.…

HotSpot 虚拟机对象探秘-对象的创建、内存布局、访问定位

目录对象的创建检查类的符号引用&#xff0c;是否执行过类的加载过程分配内存指针碰撞&#xff1a;空闲列表&#xff1a;线程安全的问题&#xff0c;对分配内存空间的动作进行同步处理——TLAB初始化虚拟机对对象进行必要的设置&#xff0c;执行构造方法对象的内存布局对象头包…

Spring、MySQL、日期、BigDecimal、集合、反射、序列化中的坑与使用指南

文章目录MySQL中的坑MySQL断开连接Mysql表字段设置为not null如何解决网络瓶颈核心流程的性能查看Spring中的坑与使用注意springboot的配置文件先后顺序定时任务不进行lombok的不适用场景Spring的Bean默认名称生成规则new出来的对象不被Spring所管理SpringBean相关的注解Spring…

Java 类和对象 详解+通俗易懂

文章目录类和对象1. 面对对象的初步认识1.1 什么是面向过程&#xff1f;什么又是面向对象&#xff1f;1.2 对象、成员变量和成员方法的关系和理解2. 类的定义和使用2.1 简单认识类2.2 类的定义格式2.3 小试身手3. 类的实例化3.1 什么是实例化3.2 类和对象的说明4. this 引用4.1…

k8s上部署seata-server集群并注册到nacos上

部署前准备 第一步&#xff1a; 创建seata-server需要的表,有现成的阿里云RDS&#xff0c;就直接在RDS上创建数据库了&#xff0c;方便后面统一管理。 具体的 SQL 参考script/server/db &#xff0c;这里使用的是 MySQL 的脚本&#xff0c;数据库名称为 seata&#xff0c;还需…

对外 API 接口,请把握这3 条原则,16 个小点

对外API接口设计 安全性 1、创建appid,appkey和appsecret 2、Token&#xff1a;令牌&#xff08;过期失效&#xff09; 3、Post请求 4、客户端IP白名单 &#xff08;可选&#xff09; 5、单个接口针对IP限流&#xff08;令牌桶限流&#xff0c;漏桶限流&#xff0c;计数器…

git如何回滚,返回到之前的记录

文章目录1.建立一个reset的测试文件&#xff0c;并连续提交。2.进行回滚测试。2.1测试,回滚到第二次提交2.1.1首先使用git log命令查看commit记录2.1.2查看结果.2.1.3回滚结果提交到远程2.2.你发现自己回滚的多了3.1撤销测试3.1建立文件&#xff0c;多次填写内容上传到git3.2撤…

顺序栈和链栈的定义和使用C语言实现(附有完整代码)

栈的基本内容&#xff1a; 无论是我们接下来要讲的栈还是后面要讲到的队列&#xff0c;他们虽然在名字上不同于我们之前的顺序表或者单链表&#xff0c;但是它们本质也是线性表&#xff0c;只是在基本操作上没有表那么“自由”。比如&#xff1a;栈只能从栈顶进行插入和删除&a…

【Linux】Linux文件权限的理解

&#x1f4ac;推荐一款模拟面试、刷题神器 、从基础到大厂面试题&#xff1a;&#x1f449;点击跳转刷题网站进行注册学习 目录 一、Shell是什么&#xff1f; 1、Shell承担用户和内核间的翻译工作 2、拒绝用户非法请求&#xff0c;保护内核 3、派生子进程执行用户指令 二…

SpringCloud

SpringCloud 三 本章知识点 3.1 项目架构演变 3.1.1 单体应用架构 部署到一个war里 部署到一个web容器里&#xff08;如tomcat&#xff09; 公用一个DB 优点: 容易测试 容易部署缺点&#xff1a; 开发效率低 代码维护难 部署不灵活&#xff08;如构建时间特别长&#xff0…

人工智能轨道交通行业周刊-第19期(2022.10.17-10.23)

本期关键词&#xff1a;首都智慧地铁、AI四小龙、文本生成视频、低光照目标检测、天窗、电务人员 1 整理涉及公众号名单 1.1 行业类 RT轨道交通中关村轨道交通产业服务平台人民铁道世界轨道交通资讯网铁路信号技术交流北京铁路轨道交通网上榜铁路视点ITS World轨道交通联盟V…

四非到保研厦大,我们还有多少路要走----技术人的保研之路

前言&#xff1a; “Hello&#xff0c;大家好我叫是Dream呀&#xff01;”在1024程序员节到来之际&#xff0c;我想和大家一起分享一下技术人生的故事----我的学长2022年从四非保研至厦大的经验和经历&#xff0c;希望大家可以耐心看完&#xff0c;收获技术力量&#xff0c;更加…

Java面试题

1、JDK 、JRE和JVM 的区别&#xff1f; JDK&#xff0c;java development kit java开发工具包&#xff0c;为java程序提供开发和运行环境JRE&#xff0c; java runtime environment java运行环境&#xff0c;为java程序提供必须的运行环境JVM&#xff0c; java virtual mach…

RayVentory以改进IT的分析,RayVentory原始数据之间轻松切换

使用KeyCloak进行SSO(单点登录)-使用单点登录解决方案&#xff0c;并直接与KeyClok交互。 特定于公司的数据丰富—使用产品所有者、价格或许可证信息等信息轻松丰富您的数据&#xff0c;以获得所需数据的综合视图。 用于更深入数据管理的新连接器-添加了许多新连接器&#xff0…

Vue3中 响应式 API ( readonly、shallowReadonly、toRaw、markRaw ) 详解

传送门&#xff1a;Vue3中 响应式 API ( reactive、ref、toRef、toRefs ) 详解 传送门&#xff1a;Vue3中 响应式 API&#xff08; shallowReactive、shallowRef、triggerRef 、customRef &#xff09;详解 1. readonly 函数 接受一个对象 (不论是响应式还是普通的) 或是一个…