【C++】初识模板

放在专栏【C++知识总结】,会持续更新,期待支持🌹


前言

在谈及本章之前,我们先来聊一聊别的。橡皮泥大家小时候应该都玩过吧,通常我们买来的橡皮泥里面都会带有一些小动物的图案的模子。我们把橡皮泥往上面按压,就会得到一个个具有该图案形状的橡皮泥。橡皮泥的颜色不同,得到的形状的颜色也不相同。就好像下面这样:

我们可以看到,我们通过给这个模子不同的材料,从而得到由不同材料铸成的不同结果,但是本身还是一个爱心,只不过材料不同。本次章节所讲解的模板,作用就类似于图纸里的这些“模子”。

泛型编程

模板的作用

  • 所谓泛型编程,实际上就是指编写与类型无关的代码,从而实现代码的复用。而模板,则是泛型编程的基础,我们可以通过模板,来实现虽然类型不同(橡皮泥颜色各异),但最终目的相同(都是得到爱心,只不过是不一样的爱心,红黄蓝绿...)。

函数模板

应用场景

这里,假如我们想要实现一个交换的函数实现所有类型都可以完成交换。我们可以采用函数重载的方式,写很多个代码,来实现int、double、char...等类型的交换,就好像下面这样:

//通过函数重载,来实现各个类型的交换
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;
}

但是,我们发现这个工作很繁琐,并且复用率低(假如出现新类型,还要再手动写)。在这种情况下,我们可以通过函数模板,来实现我们的目的。

函数模板的格式
template<typename T1, typename T2,......,typename Tn>
返回值类型 函数名(参数列表)
{}

(这里typename也可以写成class,为了保持与stl库里(SGI版本)的源码格式一致,我们采用class

上面的函数便可以写成以下这种形式:

//函数模板,实现各个类型之间的叫交换
template<class T>
void Swap(T& left, T& right)
{
    T tmp = left;
    left = right;
    right = tmp;
}

如此一来,不仅简洁了许多,避免了大量重复的书写工作,还使代码的复用率提高。接下来我们谈一谈它的原理。

函数模板的原理
  • 函数模板是一个蓝图,它本身并不是函数,编译器用使用方式产生特定具体类型函数的模具。所以其实模板就是将本来应该我们做的重复的事情交给了编译器。也就是说,在编译阶段,编译器会根据我们传入的实参的类型,来推导T的类型,从而生出来一份专有的代码。(比如我们传入的类型为int,编译器就会自动推导T的类型为int,从而实现一份int类型的代码),如下:

编译器在编译期间根据实参类型自动推演形参T的类型

函数模板实例化
  • 隐式实例化

所谓隐式实例化,就是让编译器根据实参的类型,来推演模板参数的类型。就比如:

//函数模板实例化
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;
    //隐式实例化,编译器根据a1,a2的类型,推演出T的类型为int
    Add(a1, a2);
    Add(d1, d2);
}

还有我们上面写的Swap函数,也是属于隐式实例化,由编译器来推演T的类型。

但是,假如我们这里Add(a1,d2),此时就会报错,因为编译器根据a1的类型推演T的类型为int,但是由于d2的类型为double,所以编译器推演T的类型为double,而一个T怎么可能会有两个类型,所以会报错(编译器也很懵逼,这个T到底是int,还是double???)

error

此时我们只有两种方式来解决该问题:

  1. 将其中一个参数使用类型强制转换()将类型强转为另一个参数的类型,如下:

  //类型强转
  Add(a1, (int)d2);//将d2的类型强制转换为int

不过我们一般不会采用强转的方式,而是采用接下来讲的显示实例化。(注意一点,强制类型转换会产生临时变量,临时变量具有常性,所以我们的形参得需要const来修饰

  • 显式实例化

我们可以在函数名后的<>中指定模板参数的实际类型,告诉编译器,这个T到底是啥。如下:

Add<int>(a1,d2);//告诉编译器,T的类型为int

此时如果类型不匹配,编译器会尝试进行隐式类型转换,如果无法转换成功编译器将会报错。

匹配原则
  • 假如一个非模板函数与同名的模板函数同时存在,并且模板函数可以实例化出与非模板函数相同的函数,此时编译器会调用哪一个呢?

:最优匹配的那一个,并且假如其余条件都相同,编译器会调用非模板函数。如下验证:

//匹配原则:最优原则匹配,假如条件都相同,会匹配非模板函数
//非模板函数,专门处理int类型
int Add(int left, int right)
{
    cout << "int Add(int,int)" << endl;
    return left + right;
}
//通用加法模板函数
template<class T>
T Add(T left, T right)
{
    cout << "T Add(T,T)" << endl;
    return left + right;
}
int main()
{
    Add(1, 2);//调用非模板函数
    Add<int>(1,2);//调用模板函数
    Add(3.5, 3.8);//调用模板函数
}

类模板

使用格式:
template<class T1, class T2, ..., class Tn>
class 类模板名
{
 // 类内成员定义
}; 
用法

还记得我们之前写过的栈,我们是采用typedef的形式来类型重命名,如下:

以往采用typedef来定义

这样的话,假如我们想要更换成char类型,直接在typedef处修改即可,不过这样有一个缺陷,就是我们无法同时定义一个char类型与int类型的栈。而类模板的存在则可以解决这个问题

类模板中的成员函数定义方式:

类中声明,在类外定义定义时需要加上模板参数列表。或者直接在类中定义(类中的成员函数会被当做内联函数处理,提高效率)

不过有一点需要注意,就是模板不支持声明与定义分离在不同的文件,会出现链接错误

注意事项
  • 类模板实例化需要在类模板名字后跟<>,然后将实例化的类型放在<>中即可,类模板名字不是真正的类,而实例化的结果才是真正的类。(如上:Stack只是类名Stack<int>才是类型

  • 模板不支持声明与定义分离在不同的文件,会出现链接错误!

  • 类模板在类中声明,类外定义时,需要加模板参数列表


end

生活原本沉闷,但跑起来就会有风!🌹

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【性能分析】分析JVM出现的内存泄漏的性能故障

分析JVM出现的内存持续增加的性能故障手册 前言 本文通过常见的性能文件为例&#xff0c;提供简单清晰的思路去快速定位问题根源&#xff0c;从而可以快速解决性能故障。 性能问题介绍 在性能测试工作中针对Java程序最重要的是要关注JVM的内存消耗情况&#xff0c;JVM的内存…

面试错题本

目录2023.3.21 深信服哈夫曼树哈夫曼编码2023.3.21 深信服 ​同一线程共享的有堆、全局变量、静态变量、指针&#xff0c;引用、文件等&#xff0c;而独自占有栈 友元函数不能被继承&#xff0c;友元函数不是成员函数 友元函数不能被继承&#xff0c;友元函数不是当前类的成员…

Vue2项目总结-电商后台管理系统

Vue2项目总结-电商后台管理系统 去年做的项目&#xff0c;拖了很久&#xff0c;总算是打起精力去做这个项目的总结&#xff0c;并对Vue2的相关知识进行回顾与复习 各个功能模块如果有过多重复冗杂的部分&#xff0c;将会抽取部分值得记录复习的地方进行记录 一&#xff1a;项目…

精心整理前端主流框架学习路径

版权声明 本文原创作者&#xff1a;谷哥的小弟作者博客地址&#xff1a;http://blog.csdn.net/lfdfhl 前端主流框架 前端框架指的是用于构建Web前端应用程序的框架&#xff0c;使用框架进行前端开发带来以下显著优势&#xff1a; 提高开发效率&#xff1a;前端框架提供了现成的…

STM32的CAN总线调试经验分享

相关文章 CAN总线简易入门教程 CAN总线显性电平和隐性电平详解 STM32的CAN总线调试经验分享 文章目录相关文章背景CAN总线CAN控制器CAN收发器调试过程硬件排查CAN分析仪芯片CAN控制器调试总结背景 最近负责的一个项目用的主控芯片是STM32F407IGT6&#xff0c;需要和几个电机控…

DWF文件怎么用CAD打开?DWF输入CAD步骤

DWF是一种开放、安全的文件格式&#xff0c;它可以将丰富的设计数据高效率地分发给需要查看、评审或打印这些数据的任何人。那么&#xff0c;DWF文件如何打开呢&#xff1f;下面就和小编一起来了解一下DWF输入浩辰CAD软件中的具体操作步骤吧&#xff01; DWF输入CAD中步骤&…

安装CentOS系统

打开 Oracle VM VirtualBox 点击新建 输入名称 点击下一步 点击下一步 点击创建 点击下一步 点击下一步 分配30G硬盘 点击创建 创建成功 点击启动按钮 选择 CentOS 系统 iso 镜像文件 点击启动 按键盘方向键 “上键”&#xff0c;选择第一项 按键盘回车键&#xff0c;然后等待 …

QT搭建MQTT开发环境

QT搭建MQTT开发环境 第一步、明确安装的QT版本 注意&#xff1a; 从QT5.15.0版本开始&#xff0c;官方不再提供离线版安装包&#xff0c;除非你充钱买商业版。 而在这里我使用的QT版本为5.15.2&#xff0c;在线安装了好久才弄好&#xff0c;还是建议使用离线安装的版本 在这里…

代码随想录复习——单调栈篇 每日温度 下一个更大元素12 接雨水 柱状图中最大的矩形

739.每日温度 每日温度 暴力解法双指针 def dailyTemperatures(self, temperatures: List[int]) -> List[int]:n len(temperatures)res [0] * nfor i in range(n):for j in range(i,n):if temperatures[j] < temperatures[i]: continueelse: res[i] j-ibreakreturn …

pytorch 计算混淆矩阵

混淆矩阵是评估模型结果的一种指标 用来判断分类模型的好坏 预测对了 为对角线 还可以通过矩阵的上下角发现哪些容易出错 从这个 矩阵出发 可以得到 acc &#xff01; precision recall 特异度&#xff1f; 目标检测01笔记AP mAP recall precision是什么 查全率是什么 查准率…

【K8S系列】深入解析Pod对象(一)

目录 序言 1.问题引入 1.1 问题描述 2 问题解答 2.1 pod 属性 2.1.1 NodeSelector 2.1.2 HostAliases 2.1.3 shareProcessNamespace 2.1.4 NodeName 2.1.5 其他pod属性 2.2 容器属性 2.2.1 ImagePullPolicy 2.2.2 Lifecycle 3 总结 4. 投票 序言 任何一件事情&am…

一文读懂强化学习!

一.了解强化学习1.1基本概念强化学习是考虑智能体&#xff08;Agent&#xff09;与环境&#xff08;Environment&#xff09;的交互问题&#xff1a;智能体处在一个环境中&#xff0c;每个状态为智能体对当前环境的感知&#xff1b;智能体只能通过动作来影响环境&#xff0c;当…

空间信息智能应用团队研究成果介绍及人才引进

目录1、多平台移动测量技术1.1 车载移动测量系统1.2 机载移动测量系统2、数据处理与应用技术研究2.1 点云与影像融合2.2 点云配准与拼接2.3 点云滤波与分类2.4 道路矢量地图提取2.5 道路三维自动建模2.6 道路路面三维病害分析2.7 多期点云三维变形分析2.8 地表覆盖遥感监测分析…

ChatGPT在安全研究领域的应用实践

引言ChatGPT是一个人工智能技术驱动的自然语言处理工具&#xff0c;它能够通过理解和学习人类的语言来进行对话&#xff0c;并能进行连续对话。目前ChatGPT已经官方已经更新模型到4.0版本&#xff0c;宣称它是“最先进的系统&#xff0c;能生产更安全和更有用的回复”。当前使用…

wandb:可视化和超参数寻优

参考博客&#xff1a;https://zhuanlan.zhihu.com/p/591047340 1、注册账号 首先&#xff0c;去wandb官网注册一个账号&#xff0c;选择个人使用即可&#xff08;根据个人需要&#xff09; 然后&#xff0c;登录得到一个API key 2、wandb使用 &#xff08;1&#xff09;命令…

Spring框架学习--xml和Annotation方式实现IOC

AnnotationXml的spring-IOC和全Annotation的spring-IOC 文章目录AnnotationXml的spring-IOC和全Annotation的spring-IOC学习目标第二章 基于AnnotationXml的spring-IOC【重点】1、annotationxml【入门案例】(5)【1】目标【2】实现【2.1】创建项目【2.3】改写AccountDaoImpl【2.…

刷题记录(2023.3.14 - 2023.3.18)

[第五空间 2021]EasyCleanup 临时文件包含考点 分析源码&#xff0c;两个特殊的点&#xff0c;一个是 eval&#xff0c;另一个是 include eval 经过了 strlen filter checkNums 三个函数 include 经过了 strlen filter 两个函数 filter 检测是否包含特定的关键字或字符 fun…

【数据结构与算法】用栈实现队列

文章目录&#x1f63b;前言如何用栈实现队列&#xff1f;用栈实现队列整体的实现代码&#x1f63c;写在最后&#x1f63b;前言 &#x1f61d;上一章我们用队列实现了一个栈&#xff08;-> 传送门 <-&#xff09;&#xff0c;而这一章就带大家用栈实现一个队列。 &#x1…

< 每日算法:在排序数组中查找元素的第一个和最后一个位置 >

每日算法 - JavaScript解析&#xff1a;在排序数组中查找元素的第一个和最后一个位置 一、任务描述&#xff1a;> 示例 1> 示例 2> 示例 3二、题意解析三、解决方案&#xff1a;往期内容 &#x1f4a8;一、任务描述&#xff1a; 给你一个按照非递减顺序排列的整数数组…

C++基础算法③——排序算法(选择、冒泡附完整代码)

排序算法 1、选择排序 2、冒泡排序 1、选择排序 基本思想&#xff1a;从头至尾扫描序列&#xff0c;每一趟从待排序元素中找出最小(最大)的一个元素值&#xff0c;然后与第一个元素交换值&#xff0c;接着从剩下的元素中继续这种选择和交换方式&#xff0c;最终得到一个有序…
最新文章