当前位置: 首页 > news >正文

C++好难(6):模板初阶

【本节目标】

  • 1. 泛型编程
  • 2. 函数模板
  • 3. 类模板

目录

【本节目标】

1.泛型编程

2.函数模板

概念:

格式:

原理:

实例化:

1.隐式实例化:

2.显式实例化

原则一:

原则二:

原则三:

3.类模板

格式

类模板的实例化


1.泛型编程

如何实现一个通用的交换函数呢?

以下面的交换函数为例:

// 交换两个整型变量
void Swap1(int& p1, int& p2) 
{int tmp = p1;p1 = p2;p2 = tmp;
}// 交换两个字符型变量
void Swap(char& p1, char& p2) 
{char tmp = p1;p1 = p2;p2 = tmp;
}

可以看到两种不同类型的交换函数的实现,我们用重载函数去实现的

使用函数重载虽然可以实现,但是有一下几个不好的地方:

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

如果我们像印刷一样,弄一个模具,然后让编译器根据不同的类型利用该模具来生成代码呢?

如果在C++中,也能够存在这样一个模具,通过给这个模具中填充不同材料(类型),来获得不同材料的铸件(即生成具体类型的代码),那将会节省许多头发。巧的是前人早已将树栽好,我们只需在此乘凉。

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

2.函数模板

概念:

函数模板代表了一个函数家族,该函数模板与类型无关,在使用时被参数化,根据实参类型产生函数的特定类型版本。

格式:

template<class T1, class T2, ...., class Tn>
返回值类型  函数名(参数列表)
{  }//template<typename T>
template<class T>
void Swap(T& left, T& right) {T tmp = left;left = right;right = tmp;
}

注意:typename 是用来定义模板参数关键字,也可以使用 class(切记:不能使用 struct 代替 class)

这里我们推荐使用class,因为它短,而且STL里面用的就是class

原理:

函数模板是一个蓝图,它本身并不是函数,是编译器用使用方式产生特定具体类型函数的模具。

所以其实模板就是将本来应该我们做的重复的事情交给了编译器。

在编译器编译阶段,对于模板函数的使用,编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用。

比如:当用double类型使用函数模板时编译器通过对实参类型的推演,将T确定为double类型,然后产生一份专门处理double类型的代码,对于字符类型也是如此。

实例化:

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

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

1.隐式实例化:

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

template<class T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{int a1 = 10, a2 = 20;double d1 = 1.1, d2 = 2.2;Add(a1, a2); // 编译器根据实参a1和a2推演出模板参数为int类型Add(d1, d2); // 编译器根据实参d1和d2推演出模板参数为int类型return 0;
}

但是注意以下情况属于编译失败:

因为a1是int,d1是couble,编译器无法确定此处到底该将 T 确定为 int 或者 double 类型而报错。

处理这种方式有两种方法:

1)用户自己来进行强制转换

template<class T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{int a1 = 10, a2 = 20;double d1 = 1.1, d2 = 2.2;Add((double)a1, d1); // 把a1强转成doubleAdd(a1, (int)d1); // 把d1强转成intreturn 0;
}

2)使用个显示实例化

template<class T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{int a1 = 10, a2 = 20;double d1 = 1.1, d2 = 2.2;Add<int>(a1, d1); Add<double>(a1, d1);return 0;
}

总结:这个T必须是明确的

2.显式实例化

所谓显示实例化,就是在函数名后的  < >  中指定模板参数的实际类型。

template<class T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{int a1 = 10, a2 = 20;Add<int>(a1, a1); return 0;
}

原则一:

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

// 专门处理int的加法函数
int Add(int left, int right)
{cout << "非模板调用" << endl;return left + right;
}// 通用加法函数
template<class T>
T Add(T left, T right)
{cout << "模板调用" << endl;return left + right;
}int main()
{cout << Add(1, 2) << endl; // 与非函数模板类型完全匹配,不需要函数模板实例化cout << Add<int>(1, 2) << endl; // 模板函数可以生成更加匹配的版本,编译器根据实参生成更加匹配的Add函数return 0;
}

编译器会优先去选择已经存在的函数,如果没有这个函数的存在,才会去用函数模板生成

原则二:

2)对于非模板函数和同名函数模板,如果其他条件都相同,在调动时会优先调用非模板函数而不会从该模板产生出一个实例。

如果模板可以产生一个具有更好匹配的函数, 那么将选择模板

// 专门处理int的加法函数
int Add(int left, int right)
{cout << "非模板调用" << endl;return left + right;
}// 通用加法函数
template<class T>
T Add(T left, T right)
{cout << "模板调用" << endl;return left + right;
}int main()
{cout << Add(1, 2) << endl; // 与非函数模板类型完全匹配,不需要函数模板实例化cout << Add(1.1, 2.0) << endl; // 模板函数可以生成更加匹配的版本,编译器根据实参生成更加匹配的Add函return 0;
}

原则三:

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

3.类模板

格式

类模板格式:

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

以一个栈的数据结构定义为例:

template<class T>
class Stack
{
public:Stack(int capaicty = 4):_a(new T[capaicty]),_top(0), _capacity(capaicty){}~Stack(){delete[] _a;_capacity = _top = 0;}private:T* _a;size_t _top;size_t _capacity;
};

类模板的实例化

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

// 跟上述栈定义的 代码
int main()
{Stack<int> st1; // intStack<double> st2; // doublereturn 0;
}

类模板名字不是真正的类,而实例化的结果才是真正的类

相关文章:

C++好难(6):模板初阶

【本节目标】 1. 泛型编程2. 函数模板3. 类模板 目录 【本节目标】 1.泛型编程 2.函数模板 概念&#xff1a; 格式&#xff1a; 原理&#xff1a; 实例化&#xff1a; 1.隐式实例化&#xff1a; 2.显式实例化 原则一&#xff1a; 原则二&#xff1a; 原则三&#…...

Windows 10字体模糊发虚! 如何解决?

在使用Windows 10操作系统的过程中&#xff0c;有些用户可能会遇到字体模糊、发虚的问题&#xff0c;这给用户的视觉体验带来了不小的困扰。本文将介绍几种解决Windows 10字体模糊发虚问题的方法。 一、更新显卡驱动程序 如果更新显卡驱动程序后问题仍未解决&#xff0c;那么很…...

Spring中的Bean和Bean的生命周期

在Spring中&#xff0c;Bean是被管理的对象&#xff0c;是应用程序的基本组件。Bean的生命周期包括Bean的创建、初始化、使用和销毁。在本文中&#xff0c;我们将介绍Spring中Bean的概念&#xff0c;如何创建和管理Bean以及Bean的生命周期。 Bean的概念 在Spring中&#xff0…...

嘉兴桐乡技能培训提升-如何提高工作效率

现在的工作基本上都离不开电脑&#xff0c;所以大家几乎天天都在跟电脑打交道&#xff0c;那么这些电脑使用技巧你知道多少呢&#xff1f;今天桐乡办公软件培训沈老师就给大家分享一下&#xff1a; 如何让“自动更正”输入统一的文本 你是不是经常为输入某些固定的文本&#x…...

SystemFunction032函数的免杀研究

什么是SystemFunction032函数&#xff1f; 虽然Benjamin Delphi在2013年就已经在Mimikatz中使用了它&#xff0c;但由于我之前对它的研究并不多&#xff0c;才有了下文。 这个函数能够通过RC4加密方式对内存区域进行加密/解密。例如&#xff0c;ReactOS项目的代码中显示&…...

Shell函数

目录 1.Shell的作用 2.Shell函数定义 3.函数变量的作用范围 Source 4.递归 5.函数库 ​编辑​编辑​编辑 1.Shell的作用 使用函数可以避免代码重复 使用函数可以将大的工程分割为若干小的功能模块&#xff0c;代码的可读性更强 2.Shell函数定义 1&#xff09; functi…...

Spring-IOC源码解析

容器创建过程 Spring容器的refresh方法 public void refresh() throws BeansException, IllegalStateException {synchronized (this.startupShutdownMonitor) {StartupStep contextRefresh this.applicationStartup.start("spring.context.refresh");// 1. 进行创…...

不会做大数据实时计算?10年数据分析师整理,一文给出解决方案

本文分为四个章节介绍实时计算&#xff0c;第一节介绍实时计算出现的原因及概念&#xff1b;第二节介绍实时计算的应用场景&#xff1b;第三节介绍实时计算常见的架构&#xff1b;第四节是实时数仓解决方案。 一、实时计算 实时计算一般都是针对海量数据进行的&#xff0c;并…...

如何让你的 WebSocket 接口测试更高效?拯救你的接口测试工作

目录 引言 WebSocket介绍 HTTP与WebSocket的区别 WebSocket测试方法 使用在线工具 使用Postman 使用Jmeter 使用Python 结语 引言 你是否曾经为 WebSocket 接口测试中复杂的协议和难以捕获的数据而感到束手无策&#xff1f;WebSocket 协议与传统的 HTTP 协议不同&…...

浅谈Linux 文件系统层次结构的组织方式

Linux 文件系统层次结构&#xff08;Filesystem Hierarchy Standard&#xff0c;简称 FHS&#xff09;是一种用于组织和管理 Linux 文件系统的标准化方式。该标准规定了 Linux 文件系统中各个目录和文件的组织方式、用途和权限&#xff0c;以提高文件系统的可读性、可维护性和可…...

创新案例 |探索 Tive 80% 的收入增长得益于智能物流服务、跟踪和实时可视化

您正在寻找可靠的物流解决方案吗&#xff1f; Tive 是领先的智能物流服务提供商&#xff0c;提供跟踪和实时可见性解决方案。使用 Tive&#xff0c;您可以主动监控公路、空运、海运和铁路运输。它可以帮助您减少运输问题并确保准时和全面交付&#xff0c;从而改善客户体验。 …...

makefile和cmake

Makefile 是一种文件&#xff0c;它定义了一个项目中的编译规则、依赖关系和构建过程。Makefile 可以自动化地构建和管理项目&#xff0c;使得整个项目的构建过程更加高效和可靠。下面是 Makefile 的常用语法&#xff1a; 1. 定义变量 变量可以用来保存一些常用的参数和路径&…...

通过OpenCL内核代码猜测设备寄存器个数

在OpenCL标准中&#xff0c;没有给出查看计算设备一共有多少寄存器&#xff0c;至少能分配给每个work-item多少寄存器使用的特征查询。而由于一个段内核代码是否因寄存器紧缺而导致性能严重下降也是一个比较重要的因素&#xff0c;因此我这边提供一个比较基本的方法来猜测当前计…...

C# + .Net6 实现TensorFlow图片分类

微软官网上发现一篇很有意思的文档&#xff1a;教程&#xff1a;用于对图像进行分类的 ML.NET 分类模型 - ML.NET | Microsoft Learn 这篇教程写的很学院派&#xff0c;但有点碎&#xff0c;属于上课不能打一秒钟瞌睡的那种。好在还是给出了完整的代码&#xff1a;samples/Pro…...

Ngnix负载均衡和高可用集群及搭建与相关理论

Ngnix负载均衡和高可用集群及搭建与相关理论 全文目录 Ngnix负载均衡和高可用集群及搭建与相关理论高可能保持原理配置 keepalived&#xff1a;配置keepalived的IP将外部域名解析到Keepalived的虚拟IP上如何验证配置的正确性Nginx专用调试工具ngx_conf_t如何对前后端多台服务器…...

2022年宜昌市网络搭建与应用竞赛样题(三)

网络搭建与应用竞赛样题&#xff08;三&#xff09; 技能要求 &#xff08;总分1000分&#xff09; 竞赛说明 一、竞赛内容分布 “网络搭建与应用”竞赛共分三个部分&#xff0c;其中&#xff1a; 第一部分&#xff1a;网络搭建及安全部署项目&#xff08;500分&#xff0…...

为什么PCB设计完成后需要放置mark点

PCB设计中的Mark点是指一些标记点&#xff0c;通常用于促进PCB制造和组装过程中的准确性和一致性。这些标记点在制造过程中可以帮助操作员进行自动化定位&#xff0c;从而确保所有部件都被正确组装到其正确位置&#xff0c;这对于确保产品的质量和可靠性至关重要。 下面&#…...

代理IP:IP代理技术与Socks5协议

代理IP是一种用于隐藏真实IP地址的技术&#xff0c;它可以将请求发送至代理服务器&#xff0c;再由代理服务器转发请求至目标网站。代理服务器会在请求过程中替换真实IP地址&#xff0c;从而保护用户的隐私和安全。在网络爬虫、反爬虫、匿名访问等场景中&#xff0c;代理IP技术…...

如何让java程序员生涯更顺利?我聊聊提升技术水平的五个方面

今天我想和大家聊聊程序员职业发展的问题。相信大家都知道&#xff0c;IT公司因为各种原因裁员&#xff0c;对程序员的前途发展都是不利的。特别是等到你30多岁&#xff0c;上有老下有小&#xff0c;仍然要加班&#xff0c;与年轻人竞争体力和智力&#xff0c;这是很艰难的。如…...

快速排序、希尔排序、归并排序、堆排序、插入排序、冒泡排序、选择排序(递归、非递归)C语言详解

1.排序的概念及其运用 1.1排序的概念 排序&#xff1a;所谓排序&#xff0c;就是使一串记录&#xff0c;按照其中的某个或某些关键字的大小&#xff0c;递增或递减的排列起来的操作。 稳定性&#xff1a;假定在待排序的记录序列中&#xff0c;存在多个具有相同的关键字的记录&a…...

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…...

进程地址空间(比特课总结)

一、进程地址空间 1. 环境变量 1 &#xff09;⽤户级环境变量与系统级环境变量 全局属性&#xff1a;环境变量具有全局属性&#xff0c;会被⼦进程继承。例如当bash启动⼦进程时&#xff0c;环 境变量会⾃动传递给⼦进程。 本地变量限制&#xff1a;本地变量只在当前进程(ba…...

rknn优化教程(二)

文章目录 1. 前述2. 三方库的封装2.1 xrepo中的库2.2 xrepo之外的库2.2.1 opencv2.2.2 rknnrt2.2.3 spdlog 3. rknn_engine库 1. 前述 OK&#xff0c;开始写第二篇的内容了。这篇博客主要能写一下&#xff1a; 如何给一些三方库按照xmake方式进行封装&#xff0c;供调用如何按…...

安宝特方案丨XRSOP人员作业标准化管理平台:AR智慧点检验收套件

在选煤厂、化工厂、钢铁厂等过程生产型企业&#xff0c;其生产设备的运行效率和非计划停机对工业制造效益有较大影响。 随着企业自动化和智能化建设的推进&#xff0c;需提前预防假检、错检、漏检&#xff0c;推动智慧生产运维系统数据的流动和现场赋能应用。同时&#xff0c;…...

关于iview组件中使用 table , 绑定序号分页后序号从1开始的解决方案

问题描述&#xff1a;iview使用table 中type: "index",分页之后 &#xff0c;索引还是从1开始&#xff0c;试过绑定后台返回数据的id, 这种方法可行&#xff0c;就是后台返回数据的每个页面id都不完全是按照从1开始的升序&#xff0c;因此百度了下&#xff0c;找到了…...

【磁盘】每天掌握一个Linux命令 - iostat

目录 【磁盘】每天掌握一个Linux命令 - iostat工具概述安装方式核心功能基础用法进阶操作实战案例面试题场景生产场景 注意事项 【磁盘】每天掌握一个Linux命令 - iostat 工具概述 iostat&#xff08;I/O Statistics&#xff09;是Linux系统下用于监视系统输入输出设备和CPU使…...

【快手拥抱开源】通过快手团队开源的 KwaiCoder-AutoThink-preview 解锁大语言模型的潜力

引言&#xff1a; 在人工智能快速发展的浪潮中&#xff0c;快手Kwaipilot团队推出的 KwaiCoder-AutoThink-preview 具有里程碑意义——这是首个公开的AutoThink大语言模型&#xff08;LLM&#xff09;。该模型代表着该领域的重大突破&#xff0c;通过独特方式融合思考与非思考…...

从零开始打造 OpenSTLinux 6.6 Yocto 系统(基于STM32CubeMX)(九)

设备树移植 和uboot设备树修改的内容同步到kernel将设备树stm32mp157d-stm32mp157daa1-mx.dts复制到内核源码目录下 源码修改及编译 修改arch/arm/boot/dts/st/Makefile&#xff0c;新增设备树编译 stm32mp157f-ev1-m4-examples.dtb \stm32mp157d-stm32mp157daa1-mx.dtb修改…...

微信小程序云开发平台MySQL的连接方式

注&#xff1a;微信小程序云开发平台指的是腾讯云开发 先给结论&#xff1a;微信小程序云开发平台的MySQL&#xff0c;无法通过获取数据库连接信息的方式进行连接&#xff0c;连接只能通过云开发的SDK连接&#xff0c;具体要参考官方文档&#xff1a; 为什么&#xff1f; 因为…...

uniapp中使用aixos 报错

问题&#xff1a; 在uniapp中使用aixos&#xff0c;运行后报如下错误&#xff1a; AxiosError: There is no suitable adapter to dispatch the request since : - adapter xhr is not supported by the environment - adapter http is not available in the build 解决方案&…...