C++学习之模板初阶学习
今天我们来学习C++中模板的学习。但是模板是C++中比较难的部分,因此本节我们直接出相对比较初阶的部分。
目录
泛型编程
函数模板
函数模板格式
函数模板的原理
函数模板的特性
函数模板的实例化
模板参数的匹配原则
类模板
类模板定义格式
类模板实例化
泛型编程
引例:写一个通用类型的交换函数
如果用函数重载来写,就是这样
void swap(int& a, int& b)
{int temp = a;a = b;b = temp;
}
void swap(double& a, double& b)
{double temp = a;a = b;b = temp;
}
void swap(char& a, char& b)
{char temp = a;a = b;b = temp;
}
这样当然也是可以。 但是显然我们发现,函数重载写起来十分的麻烦。每次出现一个类型都要写一个函数重载,并且很危险,一旦一个出错了所有重载都错了。
那么有没有一个办法能更简便呢?答案是有的,C++为咯应对这个情况有一个东西:模板
模板的作用就是告诉编译器一个模板,让编译器根据不同的类型利用该模板生成代码。
因此:泛型编程就是编写与类型无关的通用代码,是代码复用的一种手段。模板是泛型编程的基础。
模板分为函数模板和类模板
函数模板
函数模板代表了一个函数家族,该函数模板与类型无关,在使用时被参数化,根据实参类型产生函数的特定类型版本
函数模板格式
语法结构如下:
//模板
template<class T>
template<typename A>
注意:typename是用来定义模板参数的关键字,也可以使用class来定义(但是切记不要用struct代替class)
总结为
//模板
template<typename T1, typename T2,...... typename Tn>
返回值类型 函数名(参数列表)
{}
因此交换数据的函数我们可以这样改写:
//模板
template<typename T>
void swap(T& a, T& b)
{T temp = a;a = b;b = temp;
}
int main()
{int i = 1, j = 2;double d1 = 3.14, d2 = 2.71;char c1 = 'a', c2 = 'b';swap(i,j);swap(d1,d2);swap(c1,c2);cout << "i=" << i << " j=" << j << endl;cout << "d1=" << d1 << " d2=" << d2 << endl;cout << "c1=" << c1 << " c2=" << c2 << endl;return 0;
}
结果如下:
函数模板的原理
函数模板是一个蓝图,本身不是函数,是编译器用使用方式产生特定具体类型函数的模板,所以模板就是将本来我们要做的事情交给了编译器去做
在编译器阶段,对于模板函数的使用,编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用。例如double类型使用函数模板时,编译器通过对实参类型的推演,将T确定为double类型,然后产生一份专门处理double类型的代码。
函数模板的特性
上面的交换代码中,是不可以这么写的。
swap(i, d1); //编译错误,类型不匹配
我们可以有以下几种方法处理:类型强转和实例化
函数模板的实例化
用不同类型的参数使用函数模板的时候,成为函数模板的实例化。函数模板实例化分为隐式实例化和显式实例化
1.隐式实例化:让编译器根据实参推演模板参数的实际类型
#include <iostream>
using namespace std;
//模板
template<typename T>
void swap(const T& a, const T& b)
{T temp = a;a = b;b = temp;
}
int main()
{int i = 1, j = 2;double d1 = 3.14, d2 = 2.71;char c1 = 'a', c2 = 'b';swap(i,j);swap(d1,d2);swap(c1,c2);cout << "i=" << i << " j=" << j << endl;cout << "d1=" << d1 << " d2=" << d2 << endl;cout << "c1=" << c1 << " c2=" << c2 << endl;//隐式实例化swap((double)i, d1); return 0;
}
2.显式实例化
#include <iostream>
using namespace std;
//模板
template<typename T>
void swap(const T& a, const T& b)
{T temp = a;a = b;b = temp;
}
int main()
{int i = 1, j = 2;double d1 = 3.14, d2 = 2.71;char c1 = 'a', c2 = 'b';swap(i,j);swap(d1,d2);swap(c1,c2);cout << "i=" << i << " j=" << j << endl;cout << "d1=" << d1 << " d2=" << d2 << endl;cout << "c1=" << c1 << " c2=" << c2 << endl;//显式实例化swap<int>(i, d1); return 0;
}
模板参数的匹配原则
1.一个非模板函数可以和一个同名函数模板同时存在,而且函数模板还可以被实例化为这个非模板函数
#include <iostream>
using namespace std;
//专门的函数
void swap(int& a, int& b)
{int temp = a;a = b;b = temp;
}
//函数模板
template<typename T>
void swap(const T& a,const T& b)
{T temp = a;a = b;b = temp;
}
int main()
{int i = 1, j = 2;double d1 = 3.14, d2 = 2.71;char c1 = 'a', c2 = 'b';swap(i,j);//与非模板函数匹配,编译器无需优化swap(d1,d2);swap(c1,c2);cout << "i=" << i << " j=" << j << endl;cout << "d1=" << d1 << " d2=" << d2 << endl;cout << "c1=" << c1 << " c2=" << c2 << endl;swap(i,d1);//调用编译器优化的模板函数return 0;
}
2.对于非模板函数和同名函数模板,如果条件相同,在调用的时候会优先调用非模板函数而不是从该模板实例化一个函数。如果模板可以产生一个更好匹配的函数则选择模板
#include <iostream>
using namespace std;
//专门的函数
void swap(int& a, int& b)
{int temp = a;a = b;b = temp;
}
//函数模板
template<typename T>
void swap(const T& a,const T& b)
{T temp = a;a = b;b = temp;
}
int main()
{int i = 1, j = 2;double d1 = 3.14, d2 = 2.71;char c1 = 'a', c2 = 'b';swap(i,j);//与非模板函数匹配,编译器无需优化swap(d1,d2);swap(c1,c2);cout << "i=" << i << " j=" << j << endl;cout << "d1=" << d1 << " d2=" << d2 << endl;cout << "c1=" << c1 << " c2=" << c2 << endl;swap(i,d1);//调用编译器优化的模板函数更匹配,根据实参生成更匹配的函数return 0;
}
3.模板函数不允许自动类型转换,但是普通函数可以进行自动类型转换
类模板
类模板定义格式
template<class T1,class T2......class Tn>
class 类模板名
{//类内成员定义
}
模板不建议声明与定义分离,容易出问题。
#include<iostream>
using namespace std;
namespace MyNamespace
{template<typename T>class stack {public:stack(int n = 4):_a(new T[n]), _size(0), _capacity(n){}void print() {cout << "MyClass data: " << data << endl;}~stack(){delete[] _a;_a = nullptr;_size = 0;_capacity = 0;}private:T* a;size_t _size;size_t _capacity;};
}
int main()
{//类模板只能显式实例化MyNamespace::stack<int> s(10);MyNamespace::stack<double> d(3.14);return 0;
}
类模板实例化
类模板实例化与函数模板实例化不同。类模板实例化需要类模板名字后面加<>,然后将实例化的类型放在<>中即可,类模板名字不是真的类,而实例化的结果才是真的类。
//stack是类名,stack<int>才是类型
stack<int>st1; //int
stack<double>st2 //double
在之前写stack的时候,我们会用typedef来解决的
#include<iostream>
using namespace std;
namespace MyNamespace
{typedef int STDataType;class stack {public:stack(int n = 4):_a(new STDataType[n]), _size(0), _capacity(n){}void print() {cout << "MyClass data: " << data << endl;}~stack(){delete[] _a;_a = nullptr;_size = 0;_capacity = 0;}private:STDataType* a;size_t _size;size_t _capacity;};
}
但是typedef的原理与类模板完全不一样(后者更复杂)。typedef相当于宏定义,在预处理阶段直接替换 ,但是我们用typedef的时候没办法实例化两个类型不一样的类,因此我们需要实例化两类型的类的时候需要用类的模板
本期博客就先到这里了,模板的内容我们后续会更加深入的学习。各位读者大大求一个点赞,谢谢
相关文章:
C++学习之模板初阶学习
今天我们来学习C中模板的学习。但是模板是C中比较难的部分,因此本节我们直接出相对比较初阶的部分。 目录 泛型编程 函数模板 函数模板格式 函数模板的原理 函数模板的特性 函数模板的实例化 模板参数的匹配原则 类模板 类模板定义格式 类模板实例化 泛型…...
专业级软件卸载工具:免费使用,彻底卸载无残留!
在数字生活节奏日益加快的今天,我们的电脑就像每天都在"吃进"各种软件。但您是否注意到,那些看似消失的程序其实悄悄留下了大量冗余文件?就像厨房角落里积攒的调味瓶空罐,日积月累就会让系统变得"消化不良"。…...
JSON|cJSON 介绍以及具体项目编写
一、JSON介绍 JSON(JavaScript Object Notation 即JavaScript对象表示法)是一种轻量级的数据交换格式。采用完全独立于编程语言的文本格式来存储和表示数据。 JSON是一种数据交换格式.JSON独立于编程语言(你不必学习JavaScript).JSON表达数据的方式对通…...
Cell | 大规模 单细胞图谱 揭示非小细胞肺癌抗PD-1治疗后的免疫微环境异质性
–https://doi.org/10.1016/j.cell.2025.03.018 A single-cell atlas reveals immune heterogeneity in anti-PD-1-treated non-small cell lung cancer 留意更多内容:组学之心 研究简介 背景与问题 非小细胞肺癌(NSCLC)术后复发率高。新…...
光流 | 基于深度学习的光流估计算法汇总,原理,公式,流程图,代码
基于深度学习的光流算法 一、光流估计的基本原理二、基于深度学习的光流估计算法1. **FlowNet系列**2. **FlowNet 2.0**3. **PWC-Net**4. **RAFT(Recurrent All-Pairs Field Transformers)**5. **LiteFlowNet系列**三、算法流程图示例FlowNet2.0架构PWC-Net金字塔处理流程四、…...
常见的算法介绍
算法概述线性回归(Linear Regression)线性回归是一种通过属性的线性结合来进行预测的线性模型, 其目的是找到一条直线, 一个平面或者更高维的超平面, 使预测值和真实值之间的误差最小化逻辑回归(Logistic Regression)逻辑回归是一种分类模型, 入二分类公式 P ( Y 1 ∣ X ) e …...
【基于 LangChain 的异步天气查询1】异步调用 Open-Meteo API 查询该城市当前气温
目录 一、功能概述 二、文件结构 三、城市天气实时查询(运行代码) weather_runnable.py main.py 运行结果 四、技术亮点 五、使用场景 一、功能概述 它实现了以下主要功能: 用户输入地点(城市名) 构造提示词…...
深入解析JavaScript变量作用域:var、let、const全攻略
在JavaScript中,变量作用域是一个核心概念,它决定了变量的可访问性和生命周期。理解变量作用域对于编写清晰、高效且无错误的代码至关重要。本文将深入探讨JavaScript中不同类型的变量声明方式(var、let、const等),分析…...
C33-函数嵌套及编码实战
我们以一个编程题目的实践来学习此部分内容 题目:输入四个数,以函数的方式找出最大值 思维:使用两个数找出较大值→较大值与第三个数比较得出新的较大值→新的较大值与第四个数比较得出最大值 代码 #include <stdio.h>//内层函数的封装int GetMaxFromTwoNums(int a,int…...
clangd与clang-tidy
Clangd是基于Clang的Language Server,主要用于提供代码补全、跳转定义、错误提示等IDE功能。而Clang-Tidy则是静态代码分析工具,用于检查代码中的潜在问题,比如风格违规、潜在bug等。 clangd 核心工作原理 1. 基于编译器的精准解析 底层引擎…...
【Linux】冯诺依曼体系结构和操作系统的理解
目录 冯诺依曼体系结构一个例子来深入理解 初识操作系统操作系统的作用设计操作系统的目的操作系统之上和之下分别有啥 管理的精髓,先描述,再组织 冯诺依曼体系结构 我们知道,计算机这个东西发明出来就是帮助人们快速解决问题的。那如果我们想…...
Windows系统Jenkins企业级实战
目标 在Windows操作系统上使用Jenkins完成代码的自动拉取、编译、打包、发布工作。 实施 1.安装Java开发工具包(JDK) Jenkins是基于Java的应用程序,因此需要先安装JDK。可以从Oracle官网或OpenJDK下载适合的JDK版本。推荐java17版本&#x…...
服务预热原理
Java、Spring、Springboot工程启动后,第一次访问比较慢,而从第二次访问开始就快很多,这通常是由以下几个原因导致的: 类加载与初始化开销 类加载过程:Java程序在启动时需要加载大量的类文件到内存中,包括…...
Python核心编程深度解析:作用域、递归与匿名函数的工程实践
引言 Python作为现代编程语言的代表,其作用域管理、递归算法和匿名函数机制是构建高质量代码的核心要素。本文基于Python 3.11环境,结合工业级开发实践,深入探讨变量作用域的内在逻辑、递归算法的优化策略以及匿名函数的高效应用,…...
python环境搭建和pycharm的安装配置以及使用face_recognition与cv2
一.python环境的搭建: 1.下载python(这里以python3.11为例) step 1:打开下载网址:https://www.python.org/downloads/windows/ step 2:我这里选着python3.11.9的版本 2. 安装我就不说了,网上很多 二.pycharm的安装…...
养生:为健康生活筑牢根基
养生并非遥不可及的目标,而是贯穿于日常生活的点滴之中。从饮食、运动到心态调节,每一个环节都对我们的健康有着重要意义。以下为你详细介绍养生的实用策略,助力你开启健康生活模式。 饮食养生:科学搭配,滋养生命 合…...
linux-----------Ext系列⽂件系统(上)
1.理解硬盘 1-1 磁盘、服务器、机柜、机房 机械磁盘是计算机中唯⼀的⼀个机械设备 磁盘--- 外设 慢 容量⼤,价格便宜 1-2 磁盘物理结构 1-3 磁盘的存储结构 扇区:是磁盘存储数据的基本单位,512字节,块设备 如何定位⼀个扇区呢…...
ts装饰器
TypeScript 装饰器是一种特殊类型的声明,能够被附加到类声明、方法、访问符、属性或参数上。它本质上是一个函数,会在运行时被调用,并且被装饰的声明信息会作为参数传递给装饰器函数。 装饰器的分类 类装饰器 类装饰器作用于类构造函数&…...
未来通信中的大型人工智能模型:基础、应用与挑战的全面综述
题目:A Comprehensive Survey of Large AI Models for Future Communications: Foundations, Applications and Challenges 作者:江沸菠,潘存华,董莉,王可之,Merouane Debbah,Dusit Niyato&…...
青藏高原七大河流源区径流深、蒸散发数据集(TPRED)
时间分辨率 月空间分辨率 1km - 10km共享方式 开放获取数据大小 83.27 MB数据时间范围 1998-07-01 — 2017-12-31元数据更新时间 2024-07-22 数据集摘要 通过构建耦合积雪、冻土、冰川等冰冻圈水文物理过程的WEB-DHM模型(Water and Energy Budget-based Distribute…...
5.2 参数管理
目标 访问参数,用于调试、诊断和可视化;参数初始化;在不同模型组件间共享参数。 模型:单隐藏层的MLP import torch from torch import nnnet nn.Sequential(nn.Linear(4, 8), nn.ReLU(), nn.Linear(8, 1)) X torch.rand(size…...
Best Video下载器——抖音视频去水印工具
在浏览抖音时,我们常常会遇到一些精彩的短视频,想要保存下来作为创作素材或与朋友分享。然而直接下载的视频往往带有平台水印,影响观看体验。今天就为大家介绍几种简单实用的去水印方法,让你轻松获取高清无水印视频。 目前市面上…...
C语言_程序的段
在 C 语言程序中,内存通常被分为多个逻辑段,每个段存储不同类型的数据。理解这些段的结构和功能,有助于你更高效地编写、调试和优化程序。以下是 C 语言程序中主要的内存段及其特点: 1. 代码段(Text Segment) 存储内容:编译后的机器指令(程序代码)。特性: 只读:防止…...
Google Earth Pro(谷歌地球)2025大陆版安装教程
软件介绍 【名称】:Google Earth Pro(谷歌地球)2025 【大小】:63.6M 【语言】:简体中文 【安装环境】:Win/Win8/Winxp/Win10/Win11 谷歌地球(Google Earth) 是由Google公司开发的地图软件。谷歌地球采用了…...
2025年数维杯赛题C题专家 组委会C题专家疑集锦
1、段前段后距,行间距有要求嘛 C题专家:一般是单倍行距 2、请问参考文献和附录上方也要有图示页眉吗?ai使用报告放在附录里还是附录之后? C题专家:附录 3、第三问的那个三天都在一个城市可以吗?这样我们列两份城市的清明自由行,还是说…...
C.循环函数基础
循环函数基础 1. 循环函数基础1.1 循环的定义与作用1.1.1 `for` 循环语法示例1.1.2 `while` 循环语法示例1.1.3 `do-while` 循环语法示例1.1.4 循环的比较1.1.5 循环的应用场景2.1 for 循环语法结构执行流程示例应用场景优点缺点2.2 while 循环语法结构执行流程示例应用场景优点…...
spark-Join Key 的基数/rand函数
在数据处理中,Join Key 的基数 是指 Join Key 的唯一值的数量(也称为 Distinct Key Count)。它表示某个字段(即 Join Key)在数据集中有多少个不同的值。 1. Join Key 基数的意义 高基数:Join Key 的唯一值…...
【Oracle认证】MySQL 8.0 OCP 认证考试英文版(MySQL30 周年版)
文章目录 1、MySQL OCP考试介绍2、考试注册流程3、考试复习题库 Oracle 为庆祝 MySQL 30 周年,截止到2025.07.31 之前。所有人均可以免费考取原价245美元 (约1500)的MySQL OCP 认证。 1、MySQL OCP考试介绍 OCP考试 OCP认证是Oracle公司推…...
不同环境下运行脚本如何解决pythonpath问题
目录 问题背景: 方法一:在 Dockerfile 中设置 PYTHONPATH: 方法二: 本地脚本内动态地设置 sys.path,以确保 Python 程序在运行时能够找到项目中的模块 注意: 问题背景: 脚本在windows环境定义 然后因为…...
照片to谷歌地球/奥维地图使用指南
软件介绍 照片to谷歌地球/奥维地图是一款由WTSolutions开发的跨平台图片处理工具,能够将带有GPS信息的照片导入Google Earth(谷歌地球)或奥维地图。该软件支持Windows、Mac、iOS、Linux和Android系统,无需下载安装,直…...
visual studio 2015 安装闪退问题
参考链接: VS2012安装时启动界面一闪而过问题解决办法 visual studio 2015 安装闪退问题...
Kubernetes 使用 containerd 实现 GPU 支持及 GPU Operator 部署指南
目录 Kubernetes 使用 containerd 实现 GPU 支持及 GPU Operator 部署指南 一、为什么 containerd 是趋势? 二、目标 三、前提条件 四、方式一:containerd nvidia-container-toolkit(基础方式) 1️⃣ 安装 NVIDIA Containe…...
【typenum】 1 说明文件(README.md)
Typenum Typenum 是一个用于在编译时计算类型级数字的 Rust 库。目前支持位、无符号整数和有符号整数。 Typenum 仅依赖 libcore,因此适用于任何平台! 导入方式 虽然 typenum 分为多个模块,但它们都通过 crate 根目录重新导出,…...
Axure疑难杂症:统计分析页面引入Echarts示例动态效果
亲爱的小伙伴,在您浏览之前,烦请关注一下,在此深表感谢! Axure产品经理精品视频课已登录CSDN可点击学习https://edu.csdn.net/course/detail/40420 课程主题:统计分析页面引入Echarts示例动态效果 主要内容:echart示例引入、大小调整、数据导入 应用场景:统计分析页面…...
PyQt5基本窗口控件(QWidget)
QWidget 基础窗口控件QWidget类是所有用户界面对象的基类,所有的窗口和控件都直 接或间接继承自QWidget类。 窗口控件(Widget,简称“控件”)是在PyQt中建立界面的主要元素。在PyQt 中把没有嵌入到其他控件中的控件称为窗口&…...
双目视觉系统中,极线校正(Epipolar Rectification)与单应性矩阵/多平面单应性模型
在双目视觉系统中,极线校正(Epipolar Rectification)的目标是使左右图像的对应点位于同一水平线上,从而简化立体匹配的搜索过程。标准的双目相机标定和校正流程,其核心原理与单应性矩阵的应用方式如下: 1. …...
【部署】win10的wsl环境下调试dify的api后端服务
回到目录 标题:win10的wsl环境下,远程调试dify的api后端服务 0. 上一篇讲解web前端服务的启动方法,本篇内容是vscode的调试模式启动api后端服务。文章4000字左右,阅读时间15-20分钟。 dify官方运行环境要求较低2U4G,…...
学习黑客了解Python3的“HTTPServer“
5 分钟深入浅出理解Python3的"HTTPServer"模块 🌐 大家好!今天我们将探索Python3中的HTTPServer模块——一个简单却强大的工具,它允许你快速创建web服务器。在网络安全学习、渗透测试和CTF挑战中,这个模块常被用来搭建…...
结合 ECharts / Ant Design Blazor 构建高性能实时仪表盘
📊 结合 ECharts / Ant Design Blazor 构建高性能实时仪表盘 文章目录 📊 结合 ECharts / Ant Design Blazor 构建高性能实时仪表盘一、前言 🔍二、技术选型 🧰三、项目配置与架构 🏗️🌐 系统整体架构流程…...
刘强东杀入自动驾驶!京东注册“Joyrobotaxi”商标
又一个互联网大佬杀入汽车赛道!这一次,是京东的刘强东。5月6日,京东集团旗下公司一口气申请注册了多个名为"Joyrobotaxi"的商标,国际分类涵盖运输工具、科学仪器等核心领域。这一动作被业界普遍解读为:京东要…...
JavaScript基础-switch分支流程控制
在JavaScript编程中,switch语句提供了一种清晰、简洁的方式来实现多路分支选择。相比于多个if...else if语句,switch语句可以使代码更加易读和易于维护,尤其是在需要根据单个变量或表达式的值进行不同路径选择时尤为有用。本文将详细介绍swit…...
MySQL:视图
目录 一、什么是视图 二、视图的创建和使用 三、修改数据 (1)修改真实表 (2)修改视图 四、删除视图 五、视图的优点 一、什么是视图 视图是一个虚拟的表,它是基于一个或多个基本表或其他视图的查询结果集。视…...
05 mysql之DDL
一、SQL的四个分类 我们通常可以将 SQL 分为四类,分别是: DDL(数据定义语言)、DML(数据操作语言)、 DCL(数据控制语言)和 TCL(事务控制语言)。 DDL 用于创建…...
各国GDP变化趋势网页特效(实用动态图表)完整实例
https://download.csdn.net/download/YUJIANYUE/90803109...
【部署】win10的wsl环境下启动dify的web前端服务
回到目录 标题:win10的wsl环境下启动dify的web前端服务 0. 官网安装指引 官网有详细的源代码部署说明,参考: https://docs.dify.ai/en/getting-started/install-self-hosted/local-source-code 1. 安装nodejs 1.1. 下载和解压文件 参考:https://no…...
Veins同时打开SUMO和OMNeT++的GUI界面
进入 Veins 工程目录(即包含 sumo-launchd.py 的目录),打开终端设置 SUMO_HOME 环境变量(指向你安装的 SUMO 路径): export SUMO\_HOME/home/veins/src/sumo-1.11.0编译 Veins 工程(包含 OMNeT…...
模型 启动效应
系列文章分享模型,了解更多👉 模型_思维模型目录。刺激先行激活,后续认知更顺畅。 1 启动效应的应用 1.1 求职面试中对面试官的影响 背景:一家知名公司在招聘过程中发现,面试官对候选人的评价往往受到多种因素的影响…...
Android 使用Paging3 实现列表分页加载、下拉刷新、错误重试、筛选功能
Android 使用Paging3 实现列表加载 Paging3是Android Jetpack组件库中的分页加载库,它可以帮助开发者轻松实现列表数据的分页加载功能。本文将逐步讲解如何使用Paging3库实现一个带有加载更多、下拉刷新、错误重试、筛选功能的列表页面。 最终效果如下 加载更多、…...
SpringBoot2集成xxl-job详解
官方教程 搭建调度中心 Github Gitee 注:版本3.x开始要求Jdk17;版本2.x及以下支持Jdk1.8。如对Jdk版本有诉求,可选择接入不同版本 clone源代码执行xxl-job\doc\db\tables_xxl_job.sql # # XXL-JOB v2.4.1 # Copyright (c) 2015-present, x…...
洛图报告中的 FSHD 是什么?—— 解密九天画芯推动的三色光源显示技术
目录 一、洛图报告新焦点:FSHD 为何成为显示产业重要突破方向? (一)洛图报告核心结论:从技术突围到产业重构 二、技术解析:FSHD 如何重构显示底层逻辑? (一)物理架构…...