【C++】模板编程入门指南:零基础掌握泛型编程核心(初阶)

文章目录
- 一、泛型编程
- 二、函数模板
- 1. 函数模板的概念和格式
- 2. 函数模板的原理
- 3. 函数模板的实例化
- 隐式实例化
- 显式实例化
- 三、类模板
一、泛型编程
泛型编程就是编写与类型无关的通用代码,是代码复用的一种手段,模板是泛型编程的基础,可能不太好理解,这里我给大家举一个现实生活中的例子,我们想做很多个草莓形状的橡皮泥玩具,并且这些草莓玩具颜色不同,效果如下:
问题来了,我们该怎么解决这个问题呢?难道拿出不同颜色的橡皮泥开始一个一个捏吗?但是这样的话效率是不是很低呢?所以我们会这样想,既然这些草莓玩具的形状相同,只是颜色不同,我们是不是可以做一个草莓模具,当我们想做一个草莓玩具的时候,就可以将对应颜色的橡皮泥填充模具,最终得到这个草莓,如下:

这样我们有了模具以后,只需要使用对应颜色的橡皮泥就可以批量制作草莓了,非常高效,这就属于泛型编程的思维,大家可能还是感受不到,我们再举一个有关编程的例子,也就是使用C语言实现两个变量的交换,如下:
void Swap(int& x, int& y)
{int tmp = x;x = y;y = tmp;
}void Swap(double& x, double& y)
{double tmp = x;x = y;y = tmp;
}void Swap(char& x, char& y)
{char tmp = x;x = y;y = tmp;
}
//其它类型就不再一一列举了
我们可以发现,如果是C语言来写这种交换函数,每一种类型我们都要把这个函数套一遍,但是其实这些函数除了类型不同基本上没有区别,如果每种类型都写一次实在太麻烦了,所以在C++中引入了模板的概念,就类似于做草莓玩具的模具,通过模具可以制作非常多的草莓玩具,通过模板可以快速生成多个函数,接下来我们正式开始学习模板
二、函数模板
1. 函数模板的概念和格式
函数模板代表了一个函数家族,也就是该函数模板与类型无关,当我们使用一个类型的函数,编译器会根据这个类型产生函数的特定类型版本,可能有点不好懂,看到后面大家就明白了,我们先来看看函数模板的语法格式:
//这里的typename可以替换为class,用于声明类型T1到Tn都是类型
template<typename T1, typename T2,......,typename Tn>
//这里正常写函数的实现即可,只是把类型换成上面的T1到Tn
//注意上面的类型名可以自己取
当我们把函数中的类型换成自己定义的类型名后,只要我们在函数中使用自己定义的类型名,比如T1,那么编译器在编译时就会按需给我们生成对应类型的函数,我们直接来举一个例子,如下:
#include <iostream>
using namespace std;//这里也可以使用template<class T>,用于声明类型,可以有多个类型
template<typename T>
void Swap(T& x, T& y)//将类型替换成T
{T tmp = x;x = y;y = tmp;
}int main()
{int a = 1;int b = 2;cout << "交换前:a:" << a << "b:" << b << endl;//编译器根据需求会自动按照模板生成一个int类型的SwapSwap(a, b);cout << "交换前:a:" << a << "b:" << b << endl;double c = 1.2;double d = 23.1;cout << "交换前:c:" << c << "d:" << d << endl;//编译器根据需求会自动按照模板生成一个double类型的SwapSwap(c, d);cout << "交换前:c:" << c << "d:" << d << endl;return 0;
}
在上面的例子中,我们使用模板写了一个Swap函数,属于函数模板,编译器按需实例化出一个又一个的Swap函数,比如我们传int类型的参数,编译器就会生成int类型的Swap函数,就跟我们上面讲的草莓玩具的例子一样,根据模具就能很轻松的制作出来草莓玩具
编译器也可以根据函数模板很轻松地实例化出不同的类型的函数,我们来看看上面举例的代码的运行结果,看看是否能做到我说的效果:
可以看到代码没有问题,这里我们简单了解了一下函数模板的格式并且举了一些例子,接下来我们详细讲讲函数模板的原理和实例化
2. 函数模板的原理
函数模板的原理其实我们都已经说过了,跟草莓模具的描述差不多,但是我们还是简单总结一下,函数模板就像是一个蓝图,它本身并不是函数,是编译器用使用方式产生特定具体类型函数的模具。所以其实模板就是将本来应该我们做的重复的事情交给了编译器,如下图:

在编译器编译阶段,对于模板函数的使用,编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用。比如:当用double类型使用函数模板时,编译器通过对实参类型的推演,将T确定为double类型,然后产生一份专门处理double类型的代码,对于其它类型也是如此
3. 函数模板的实例化
用不同类型的参数使用函数模板时,称为函数模板的实例化。模板参数实例化分为:隐式实例化和显式实例化,接下来我们一个一个介绍:
隐式实例化
隐式实例化就是不需要程序员指定相应的类型,让编译器自己去根据实参的类型实例化出对应的函数,比如之前我们举的Swap函数的例子,我们使用的就是隐式实例化,因为我们没有指定函数内部的类型,是靠编译器根据实参实例化出的
隐式实例化有时候非常方便,因为不需要我们关心类型,但是隐式实例化也有它的不足,因为在一定的情况下编译器也不知道到底怎么实例化出对应的函数,我们来看以下的例子:
#include <iostream>
using namespace std;template<class T>//也可以用typename
T Add(constt T& x, const T& y)
{return x + y;
}int main()
{int a = 1;double b = 2.2;Add(a, b);return 0;
}
在上面的例子中,我们设计Add的函数模板时只给了一个类型,但是传参数时却传了两种类型,编译器此时就不知道到底该用哪种类型去实例化,所以就会报错,解决办法有三种,第一种方法就是对任意一个参数进行强制类型转换,让参数类型强行一致,第二种方法就是增加模板参数,第三种方法就是显式实例化
我们先讲一下前两种方法,最后一种方法我们在显式实例化部分讲解,如下:
//方法一:将某个变量进行强转
Add(a, (int)b);//方法二:增加模板参数
template<class T1, class T2>//也可以用typename
T1 Add(constt T1& x, const T2& y)
{return x + y;
}
上面两种方法都能解决报错,但是给我们的感觉就是麻烦,因为如果一个函数有很多参数,就有可能强转多次,并且修改模板参数也很麻烦,最后还是只能返回一种类型,跟强转效果差不多但是更麻烦,那么有没有什么更好的方法呢?其实就是第三种方法,显式实例化,我们一起来学习
显式实例化
显式实例化就是在函数后面加一对尖括号,里面写上类型,这样就说明你已经确定要将这个函数实例化为某种类型,不用编译器去推导,过程中如果出现其它类型,那么编译器就会将它强转为尖括号中的类型,如果强转不了就会报错,如下:
//方法三:显式实例化
Add<int>(a, b);//b会被强转为整形
一般来说显式实例化用在类模板实例化中,但是其实函数模板也可以使用,就是用来应对上面的这种场景,那么函数模板我们就了解到这里,后面我们还有进阶,不过要等后面讲完STL后我们再学习,接下来我们就来学习类模板
三、类模板
有了函数模板的基础我们学习类模板就轻松很多了,因为类模板和函数模板类似,甚至比函数模板更简单,我们先来看看类模板的格式,如下:
//这里可以是class也可以是typename
template<class T1, class T2, ..., class Tn>
class 类模板名
{// 类内成员定义以及成员函数的实现
};
可以看出类模板的定义和函数模板的定义都是非常相似的,是在类的最前面定义好类型供我们在实现中使用,接下来我们就来举一个例子,看看类模板是否实用,如下:
#include<iostream>
using namespace std;
// 类模版
template<typename T>
class Stack
{
public:Stack(size_t capacity = 4){_array = new T[capacity];_capacity = capacity;_size = 0;}private:T* _arr;size_t _capacity;size_t _size;
};
这里我们利用类模板简单写了一个栈,跟之前唯一不同的是其中数据类型被替换成我们定义的类型T了,这样写的好处是可以根据要求实例化出不同类型的栈,比typedef一个类型好用多了,因为类模板可以同时使用多个类型的栈,但是typedef做不到,它一次只能替换一个类型,使用类模板是非常重要的
除此之外类模板的实例化也很重要,它不能由编译器来自动识别,也就是不能隐式实例化, 只能显式实例化,这是为什么呢?我们来看看一个类对象的实例化语句,如下:
Stack st;
不知道大家发现没有,类对象实例化时是看不出来类型的,函数模板可以通过实参推导,可类模板做不到,因为没有参数,所以类模板只能采用显式实例化的方式实例化对象,如下:
//让类模板实例化出来一个int类型的栈
Stack<int> st1;
//让类模板实例化出来一个double类型的栈
Stack<double> st1;
那么今天模板初阶的内容就讲到这里,如果有问题欢迎在评论区提出,后面我们就可以开始STL的学习了,敬请期待吧!
bye~
相关文章:
【C++】模板编程入门指南:零基础掌握泛型编程核心(初阶)
文章目录 一、泛型编程二、函数模板1. 函数模板的概念和格式2. 函数模板的原理3. 函数模板的实例化隐式实例化显式实例化 三、类模板 一、泛型编程 泛型编程就是编写与类型无关的通用代码,是代码复用的一种手段,模板是泛型编程的基础,可能不太…...
React实现lottie文件预览(可识别json文件或压缩包带资源的素材)
React实现lottie文件预览(可识别json文件或压缩包带资源的素材) 🔴 1、React实现lottie文件预览,所用到的第三方库 🟢 1.1、 react-lottie jszip-syncnpm install react-lottie jszip-sync // 或者yarn add react-…...
网上打印平台哪个好用?网上打印资料推荐
网上打印平台哪个好用 随着数字化办公的普及,网上打印平台因其便捷性和经济性而受到越来越多人的青睐。无论是学生、上班族还是个人用户,在需要快速打印资料时,一个好用的在线打印服务可以大大节省时间和成本。 那么,如何选择一…...
Mac远程桌面软件哪个好用?
远程桌面软件能帮助我们快速的远程控制另一台电脑,从而提供远程帮助,或者进行远程办公。那么,对macOS系统有什么好用的Mac远程桌面软件呢? 远程看看是一款操作简单、界面简洁的远程桌面软件,支持跨平台操作࿰…...
【回溯 力扣】17. 电话号码的字母组合
题目 17. 电话号码的字母组合 思路 定义数组存储数字对应的字符串,本题回溯时为index1,因为下一个数字选的是下一个字符串,前两题都是属于同一个字符串。 代码 class Solution { private:vector<string>result;string duiying[10]{"&quo…...
【基础1】冒泡排序
核心思想 冒泡排序是通过相邻元素的连续比较和交换,使得较大的元素逐渐"浮"到数组的末尾,如同水中气泡上浮的过程 特点: 每轮遍历将最大的未排序元素移动到正确位置稳定排序:相等元素的相对位置保持不变原地排序…...
C#—Settings配置详解
C#—Settings配置详解 在C#项目中,全局配置通常指的是应用程序的设置(settings),这些设置可以跨多个类或组件使用,并且通常用于存储应用程序的配置信息,如数据库连接字符串、用户偏好设置等。 Settings配置…...
详解DeepSeek模型底层原理及和ChatGPT区别点
一、DeepSeek大模型原理 架构基础 DeepSeek基于Transformer架构,Transformer架构主要由编码器和解码器组成,在自然语言处理任务中,通常使用的是Transformer的解码器部分。它的核心是自注意力机制(Self - Attention),这个机制允许模型在处理输入序列时,关注序列中不同位…...
PyCharm中通过命令行执行`pip`命令下载到哪里了:虚拟环境目录下
PyCharm中通过命令行执行pip命令下载到哪里了:虚拟环境目录下 在PyCharm中通过命令行执行pip命令安装工具包,包的下载位置取决于多种因素 虚拟环境 如果项目使用了虚拟环境(通常是推荐的做法): Windows:虚拟环境通常位于项目目录下的.venv文件夹(默认情况)或你指定…...
Golang的性能分析指标解读
Golang的性能分析指标解读 一、概述 语言)是一种由Google开发的开源编程语言,以其并发性能和高效的编译速度而闻名。对于程序员来说,了解如何对Golang应用程序进行性能分析是非常重要的,因为这能帮助他们发现潜在的性能瓶颈并对其…...
QT 作业 day4
作业 代码 Widget.h class Widget : public QWidget {Q_OBJECTpublic:Widget(QWidget *parent nullptr);~Widget();private slots:// 槽函数void on_listWidget_itemDoubleClicked(QListWidgetItem *item);private:Ui::Widget *ui; }; #endif Widget.cpp Widget::Widget(QW…...
力扣刷题——4.寻找两个正序数组的中位数
题目要求在两个有序数组中找到中位数。由于时间复杂度要求为 O(log(mn)),因此不能简单地将两个数组合并后再找中位数,而是需要用二分查找的思路来解决。 解决思路:二分查找 将问题转化为在两个有序数组中寻找第 k小的数,其中 k 是…...
redis 与 DB 的一致性 7 种策略
为什么要使用 redis 做缓存?封底估算为什么是单行数据的QPS,而不是总的? 什么时候使用DB,Redis,本地缓存 数据的分类一致性的方案1. 先清除Redis,再更新 DB2. 先更新DB,再清除 Redis使用场景: 3. 延迟删除与延迟双删使用场景 4. 监听 binlog 清除5. 双写使用场景: 6. 监听bin…...
Docker安装Redpandata-console控制台
介绍 Redpanda控制台,这是一个功能强大的Web UI,用于管理和监控您的Redpanda 集群。探索实际示例和场景,以帮助您了解如何利用 Redpanda 控制台实现不同的用例,包括数据可观察性、Redpanda 管理、访问控制和连接。 可对Redpanda…...
【自学笔记】NoSQL基础知识点总览-持续更新
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 NoSQL基础知识点总览一、NoSQL简介二、NoSQL数据库类型三、NoSQL数据库特点四、MongoDB基础示例1. 安装MongoDB2. 启动MongoDB服务3. 使用MongoDB Shell4. 创建数据…...
专业 英语
文章目录 1.计算机(1)计算机组成原理(2)计算机网络(3)数据库(4)编译原理(5)编程词汇(6)开发术语(7)芯片(8)嵌入式硬件(9)职场(10)软件(11)论文(12)深度学习 DL(13)计算机视觉 CV(14)自动驾驶(15)自然语言处理 NLP(16)计算机图形学(17)Linux 2.数学3.机械、材料5.医药/护肤6.奢侈…...
【分享】网间数据摆渡系统,如何打破传输瓶颈,实现安全流转?
在数字化浪潮中,企业对数据安全愈发重视,网络隔离成为保护核心数据的重要手段。内外网隔离、办公网与研发网隔离等措施,虽为数据筑牢了防线,却也给数据传输带来了诸多难题。传统的数据传输方式在安全性、效率、管理等方面暴露出明…...
Docker创建自定义网桥并指定网段
前言 docker0是Docker默认网络的核心组件, 通过虚拟网桥和NAT技术, 实现了容器间的通信以及容器与外部网络的交互。然而, docker0网段是固定的(通常是172.17.0.0/16), 为了更灵活地管理容器网络,Docker支持创建自定义网桥,允许用户指定网段。 例如, 在…...
【蓝桥】常用库函数
1、memset()函数 1.1 基本介绍 定义在头文件<cstring>中主要作用是对一块内存区域进行设置值的操作 1.2 函数原型 void *memset(void *str, int c, size_t n);str:指向要填充的内存块的指针c:要设置的值。该值以int形式传递,但函数在…...
用DeepSeek生成批量删除处理 PDF第一页工具
安装依赖库 在运行程序之前,请确保安装所需的库: pip install pymupdf python-docx Python 程序代码 import os import fitz # PyMuPDF from docx import Documentdef delete_pdf_first_page(input_path, output_path):"""删除 PDF…...
关于高精度力扣66
class Solution { public: vector<int> plusOne(vector<int>& d) { if(d.back()<9){ d[d.size()-1]; return d; } else{ string temp; for(int i0;i<d.size();i){ temptempto_string(d[i]); } int astoi(temp);//这里会报错是因为超过int表示的范围了 a…...
03.03 QT
1.在注册登录的练习里面,追加一个QListwidget 项目列表 要求:点击注册之后,将账号显示到 1istwidget上面去 以及,在listwidget中双击某个账号的时候,将该账号删除 Widget.h: #ifndef WIDGET_H #define WIDGET_H#include <QWi…...
华为云 | 快速搭建DeepSeek推理系统
DeepSeek(深度求索)作为一款国产AI大模型,凭借其高性能、低成本和多模态融合能力,在人工智能领域崛起,并在多个行业中展现出广泛的应用潜力。 如上所示,在华为云解决方案实践中,华为云提供的快速…...
BUUCTF [BJDCTF2020]EasySearch1
写一篇文章来学习一下 ssi 注入 以及 dirmap 工具的使用 看到这两个框框没什么想法,边探索边扫下目录吧。显示前端报错,先禁用了js,然后又尝试抓了下包,没有发现什么,只好看看扫出来的目录了,最终扫出来了…...
探秘基带算法:从原理到5G时代的通信变革【五】CORDIC算法
文章目录 2.4 CORDIC算法2.4.1 CORDIC算法的基本原理2.4.2 方法论与分类体系旋转模式矢量模式线性模式 2.4.3 **CORDIC 算法中的误差来源****角度逼近误差的分析****缩放效应误差的分析****精度需求与迭代次数的关系****常见应用场景下的迭代次数建议****总结** 2.4.4优缺点分析…...
FPGA学习篇——Verilog学习2
1 系统函数 Verilog 语言中预先定义了一些任务和函数,用于完成一些特殊的功能,它们被称为系统任务和系统函数,这些函数大多数都是只能在 Testbench 仿真中使用的,使我们更方便的进行验证。 1.1 时间预编译指令及延时 时间精度&a…...
51单片机编程学习笔记——74HC245八路三态输出双向收发器
大纲 组成电源引脚和地引脚使能输入端DIR(T/R)引脚A端和B端 工作原理数据传输方向控制使能控制 在单片机系统里,单片机的 I/O 口驱动能力往往有限。当需要连接较多外部设备或者负载较大时,就可能出现信号传输不稳定的问题。74HC24…...
C++:类和对象(下篇)
1. 再谈构造函数 1.1 构造函数体赋值 在创建对象时,编译器通过调用构造函数,给对象中各个成员变量一个合适的初始值。 class Date { public:Date(int year, int month, int day){_year year;_month month;_day day;} private:int _year;int _mont…...
Full GC 排查
在 Java 中,Full GC(完全垃圾回收)会对整个堆(包括年轻代和老年代,甚至可能包括永久代/元空间)进行垃圾回收,通常会导致较长的停顿(STW,Stop-The-World)。如果…...
DeepSeek集成到VScode工具,让编程更高效
DeepSeek与VScode的强强联合,为编程效率树立了新标杆。 DeepSeek,一款卓越的代码搜索引擎,以其精准的索引和高速的检索能力,助力开发者在浩瀚的代码海洋中迅速定位关键信息。 集成至VScode后,开发者无需离开熟悉的编辑…...

