C++ 函数模板和类模板
参考视频:C++类模板_哔哩哔哩_bilibili
遗留问题:编译器怎么处理函数模板和类模板
目录
一、为什么会有函数模版?函数模板是为了解决什么问题?
二、函数模板的概念
三、函数模版的使用
四、函数模板的特化
五、类模板的概念
六、类模板的使用
七、类模板的特化
八、类模板的偏特化
一、为什么会有函数模版?函数模板是为了解决什么问题?
函数模板是为了减少重复代码,是整合多个函数的函数体相同而区别只在参数类型不同的代码冗余的情况。不用函数模板前,多个函数的参数类型不同,但函数体相同,导致重复代码较多。有了函数模板,只需要一份函数体一份代码就可以实现原来需要几个函数体才实现的功能,例如不使用函数模板时,要分别对int、char等类型的两个数据做是否相等判断,需要定义多个函数:
bool isEqual( int nA, int nB)
{return nA == nB;
}bool isEqual(char cA, char cB)
{return cA == cB;
}int main()
{int nA = 10;int nB = 20;char cA = 'A';char cB = 'B';std::cout << isEqual(nA, nB)<< std::endl;std::cout << isEqual(cA, cB) << std::endl;return 0;
}
可见重载的两个函数的函数体实现一样,函数体代码重复,下面是使用函数模板的代码:
#include <iostream>
//使用模板代替原来的多个函数
template<typename T>
bool isEqual(T tA, T tB)
{return tA == tB;
}int main()
{
//变量以匈牙利命名法命名int nA = 10;int nB = 20;char cA = 'A';char cB = 'B';
//编译期间,编译器根据实参类型和模板函数生成函数bool isEqual(int, int);std::cout << isEqual(nA, nB)<< std::endl;
//编译期间,编译器根据实参类型和模板函数生成函数bool isEqual(char, char);std::cout << isEqual(cA, cB) << std::endl;return 0;
}
二、函数模板的概念
函数模板是用于生成函数的模板。在编译阶段,编译器会根据函数模板的使用情况(例如上面的isEqual(nA, nB)、isEqual(cA, cB)创建出函数名相同,参数类型由编译器判断的若干函数。通过函数模板创建的函数拥有相同的函数体,只是函数的参数类型不同。
三、函数模版的使用
每当在一个编译单元(经过预处理的.cpp文件)中使用了函数模版,则必须在该编译单元中给出函数模版的定义(要是把函数模板声明写在.h文件,函数模板声明写在.cpp文件,然后在使用函数模板时只使用include 包含函数模板的头文件,不给出函数模板的定义,那么编译会报无法解析的外部符号错误)。因此,为了避免在每个编译单元都定义函数模板,建议在头文件中对函数模板进行声明定义(即将函数模板的声明和定义都写在一个头文件中,然后在需要使用函数模板的地方加上该头文件)。
函数模版的声明:
template<typename T>返回类型 函数名(参数列表);其中T表示任意类型,参数的类型和返回类型都可以指定为T
函数模版的定义
template<typename T,..> //可指定多个泛型
返回类型 函数名(参数列表)
{//函数体
}
将一、为什么会有函数模版?函数模板是为了解决什么问题?这一节中的函数模板的声明和定义写在isEqualFunc.h头文件中再使用:
#ifndef ISEQUALFUNC_H
#define ISEQUALFUNC_H
//函数模板声明
template<typename T>
bool isEqual(T tA, T tB);//函数模板定义
template<typename T>
bool isEqual(T tA, T tB)
{return tA == tB;
}#endif // ISEQUALFUNC_H
//这是main.cpp
#include "isEqualFunc.h"
#include <iostream>int main()
{//变量以匈牙利命名法命名int nA = 100111111;int nB = 111111;char cA = 'A';char cB = 'A';//编译期间,编译器根据实参类型和模板函数生成函数bool isEqual(int, int);std::cout << isEqual(nA, nB)<< std::endl;//编译期间,编译器根据实参类型和模板函数生成函数bool isEqual(char, char);std::cout << isEqual(cA, cB) << std::endl;return 0;
}
四、函数模板的特化
函数模板特化是指在实例化函数模板时,对特定类型的实参进行特殊的处理,即当实参为特定类型时,通过函数模板生成的函数会有不同的函数体。
特化时需要为函数模板添加新的定义,方式如下:
template<>
返回类型 函数名<特定的参数类型>(参数列表)
{//函数体
}
在上面的例子中,对于char*类型的字符串,使用上面的函数模板无法对字符串正确判断是否相等,因为字符串比较不是直接比较指针是否相等,而要调用strcmp函数判断,对上面的例子就需要对函数模板进行特化
#ifndef ISEQUALFUNC_H
#define ISEQUALFUNC_H
#include <string.h>
//函数模板声明
template<typename T>
bool isEqual(T tA, T tB);//函数模板定义
template<typename T>
bool isEqual(T tA, T tB)
{return tA == tB;
}//函数模板的特化
template<>
bool isEqual< char* >( char* szA, char* szB)
{return strcmp(szA, szB) == 0;
}#endif // ISEQUALFUNC_H
//这是main.cpp
#include "isEqualFunc.h"
#include <iostream>int main()
{//变量以匈牙利命名法命名int nA = 100111111;int nB = 111111;char cA = 'A';char cB = 'A';char szA[4] = "AAA";char szB[4] = "AAA";// 编译器会怎么处理函数模板和特化?待参考https://blog.csdn.net/zgaoq/article/details/85232968//编译期间,编译器根据实参类型和模板函数生成函数bool isEqual(int, int);std::cout << isEqual(nA, nB)<< std::endl;//编译期间,编译器根据实参类型和模板函数生成函数bool isEqual(char, char);std::cout << isEqual(cA, cB) << std::endl;std::cout << isEqual(szA, szB) << std::endl; //判断字符串是否相等return 0;
}
五、类模板的概念
类模板是用于生成类的模板。在编译阶段,编译器会根据类模板的使用情况创建出仅部分成员数据类型和部分成员函数的参数类型不同,其他完全相同的若干类。
通过类模板的这些特性我们可以尝试写出用于存放不同类型数据的容器。
六、类模板的使用
类模板的声明格式:
template<typename T,...>
class 类名
{//成员
};
类模板的成员函数定义如下,一般类模板的定义也应该写在头文件中:
template<typename T,...>
返回类型 类名<T,...>::成员函数名(形参列表)
{//函数体
}
下面是一个例子,这个例子定义了一个MyArray的类模板,其中数据成员是数组,通过泛型T达到可定义各种类型数组的目的,而所有对这个数组的操作函数都相同,这样就不用针对不同类型的数组,定义不同的类,达到了复用代码即减少重复代码的目的:
//这是myArray.h
#ifndef MYARRAY_H
#define MYARRAY_H
#include <iostream>
constexpr int m_nMaxLen = 20;template <typename T>
class MyArray
{
public:MyArray():size(0){}//添加数据bool addData(T value);//删除数据bool deleteData(T value);//获取数据T getData(int nIndex);void print(){for(int i = 0; i < size; i++){std::cout << data[i] << std::endl;}}
private:T data[m_nMaxLen];int size;
};template<typename T>
bool MyArray<T>::addData(T value)
{if(size == m_nMaxLen){return false;}data[size] = value;size++;return true;
}template<typename T>
bool MyArray<T>::deleteData(T value)
{if(size == 0){return false;}int nDataIndex = -1;for(int i = 0; i < size; i++){if(data[i] == value){nDataIndex = i;break;}}if(nDataIndex == -1){std::cout<< "找不到该数据!" << std::endl;return false;}else{for(int i = nDataIndex; i < size -1; i++){data[i] = data[i+1]; //往前移动后面的数据,即可以删除该数据}size --;}return true;
}template<typename T>
T MyArray<T>::getData(int nIndex)
{if(nIndex >= 0 && nIndex < size){return data[nIndex];}else{return (T)0;}
}#endif // MYARRAY_H
//这是main.cpp
#include "myarray.h"
#include <iostream>
int main()
{std::cout <<"----int begin----" << std::endl;MyArray<int> intData;for(int i = 1; i <= 10; i++){intData.addData(i);}intData.print();std::cout <<"--------" << std::endl;intData.deleteData(11);std::cout <<"--------" << std::endl;intData.print();std::cout << intData.getData(0) << std::endl;std::cout <<"----int end----" << std::endl;std::cout <<"----char begin----" << std::endl;MyArray<char> charData;for(int i = 1; i <= 10; i++){charData.addData('A'+ i);}charData.print();std::cout <<"--------" << std::endl;charData.deleteData('B');std::cout <<"--------" << std::endl;charData.print();std::cout << charData.getData(1) << std::endl;std::cout <<"----char end----" << std::endl;return 0;
}
七、类模板的特化
类模板的特化是在实例化类模板时,对特定类型的泛型进行特殊的处理,即用户指定所有特定类型的类模板时,通过特化类模板生成的类可能与其他类有完全不同的结构。
特化类模板是需要对整个类模板进行声明定义:
template<>
class 类名<指定类型,指定类型,...>
{
//类成员
};
这里对第六节中的例子添加特化:
#ifndef MYARRAY_H
#define MYARRAY_H//....其他代码不变//对特定泛型的类的成员函数进行特化
template<>
MyArray<char>::MyArray()
{std::cout<< "MyArray<char>::MyArray()" << std::endl;
}//对特定泛型的类进行特化
//这里例子可能不太恰当,只是展示如何对特定类型的类进行特化
template<>
class MyArray<float>
{public:MyArray();private:float data;
};MyArray<float>::MyArray()
{std::cout<< "MyArray<float>::MyArray()" << std::endl;
}#endif // MYARRAY_H
八、类模板的偏特化
偏特化和特化类似,只是特化会指定所有的泛型,而偏特化只指定部分泛型。
偏特化类模板需要对整个类模板进行声明定义:
template <typename T,...不需特化的泛型...>
class 类名<指定类型,...不需特化的泛型名,...>
{//类成员;
}
例子:
#ifndef PAIR_H
#define PAIR_H
#include <iostream>
template<typename T1, typename T2>
class Pair
{
public:Pair();
private:T1 m_first;T2 m_second;
};template<typename T1, typename T2>
Pair<T1, T2>::Pair()
{std::cout << "Pair<T1, T2>::Pair()" << std::endl;
}//偏特化
template<typename T2>
class Pair<char, T2>
{
public:Pair();
private:char first;T2 second;
};template<typename T2>
Pair<char, T2>::Pair()
{std::cout << "Pair<char, T2>::Pair()" << std::endl;
}#endif // PAIR_H
//这是main.cpp
#include <iostream>
#include "Pair.h"
int main()
{Pair<int, int> pNn;Pair<char, int> pCn;return 0;
}
运行结果:

相关文章:
C++ 函数模板和类模板
参考视频:C类模板_哔哩哔哩_bilibili 遗留问题:编译器怎么处理函数模板和类模板 目录 一、为什么会有函数模版?函数模板是为了解决什么问题? 二、函数模板的概念 三、函数模版的使用 四、函数模板的特化 五、类模板的概念 …...
安卓Termux系统设备安装内网穿透工具实现远程使用SFTP传输文件
文章目录 前言1. 安装openSSH2. 安装cpolar3. 远程SFTP连接配置4. 远程SFTP访问4. 配置固定远程连接地址 前言 本教程主要介绍如何在安卓 Termux 系统中使用 SFTP 文件传输,并结合cpolar内网穿透工具生成公网地址,轻松实现无公网IP环境远程传输…...
文件属性获取
1、getpwuid函数 #include <stdio.h> #include <sys/types.h> #include <pwd.h> int main(int argc, char *argv[]) {uid_t uid 1000;struct passwd * pw getpwuid(uid);printf("name:%s gid:%d info:%s wd:%s shell:%s\n",pw->pw_name,pw-&g…...
C:冒泡排序
1、冒泡排序介绍: 冒泡排序的核心思想就是:两两相邻的元素进行比较。 先用一个例子来帮助大家理解一下冒泡排序的算法是怎们进行的 有一排高矮不同的人站成一列,要按照从矮到高的顺序重新排队。 冒泡排序的方法就是,从第一个人…...
探秘C# LINQ元素运算:原理阐释与实践指南
文章目录 一、LINQ元素运算符概述二. ElementAt 和 ElementAtOrDefault三. First 和 FirstOrDefault四. Last 和 LastOrDefault五. Single 和 SingleOrDefault六. Where 和 Select七、实际应用场景示例总结 LINQ(Language-Integrated Query)是C#中强大且…...
根据bean的名称获取bean,静态方法查询数据库
根据bean名称获取bean 1.先创建bean,如template package com.test.game.config;import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.jdbc.core.JdbcTemplate;import…...
剪画小程序:音频剪辑新手入门:基础操作指南!
亲爱的小伙伴们,你是否对音频剪辑充满好奇,却不知道从何下手?今天,就让我们用【剪画】一起揭开音频剪辑基础操作的神秘面纱! 音频拼接——打造个性音乐串烧 音频拼接是将多个音频片段组合在一起,创造出独特…...
IDEA中maven jar下载失败问题处理
前言 对于在IDEA中maven爆红问题,主要分为两类,一个是全部爆红,一个是部分爆红。 全部爆红 这类问题主要是maven配置没有搞好,可以根据下列步骤自查 1.配置好国内的Maven源 第一步:打开IDEA,查看配置 第二步: 检查…...
C++中,函数返回const类型有什么作用,请举例说明
在C中,函数返回const类型的主要作用是确保返回的对象不会被修改。这种保护机制增强了代码的健壮性和可维护性,尤其是在你希望保证函数返回的数据不被意外篡改时。下面通过几个例子来说明函数返回const类型的作用。 例子 1: 返回常量引用 当你从函数中返…...
Html详解——Vue基础
HTML是什么? 超文本标记语言(英语:HyperText Markup Language,简称:HTML)是一种用来结构化 Web 网页及其内容的标记语言。网页内容可以是:一组段落、一个重点信息列表、也可以含有图片和数据表…...
【安规电容知识点总结】
安规电容知识点总结 安规电容简介安规电容的种类X电容和Y电容X电容和Y电容示意图安规电容的型号与应用安规电容简介 安规电容:是指用于这样的场合,即电容器失效后,不会导致电击,不会危及到人身安全。 具体来说,所谓安规电容是一种与普通电容相比符合安全认证的电容,故称…...
R9000P 双系统安装 win11 和 ubuntu
网上了解到一堆关于 r9000p 安装较老的ubuntu系统,会有一堆问题 可能是电脑硬件比较新,较老的系统相关方面不兼容 那么干脆直接装新一点的系统 我安装了 Ubuntu 22.04 1 根据相关教程利用u盘制作系统盘 ultraISO 推荐使用清华源 速度快一点 https://…...
8月8日笔记
8月8日笔记 msf常见命令 启动MSF控制台 msfconsole: 启动MSF控制台。msfconsole -h: 显示帮助菜单。msfconsole -q: 启动MSF控制台并立即退出。 导航和管理 back: 返回上一级菜单。exit: 退出MSF控制台。banner: 显示MSF的横幅。cd: 更改工作目录。color: 开启或关闭彩色输…...
【单片机开发软件】使用VSCode开发STM32环境搭建
💌 所属专栏:【单片机开发软件技巧】 😀 作 者: 于晓超 🚀 个人简介:嵌入式工程师,专注嵌入式领域基础和实战分享 ,欢迎咨询! 💖 欢迎大家࿱…...
第十五届蓝桥杯大赛青少组——赛前解析(算法)
算法:进制转换、模拟算法,枚举算法,冒泡排序,插入排序,选择排序,递推算法,递归算法,贪心算法。 1.进制转换 二进制:只包含0和1 八进制:只包含0-7 十进制&…...
工作助手C#研究笔记(5)
通过示例对C#程序的结构逻辑进行研究梳理,虽然通过阅读相关书籍,但是来的效果更慢。一下相关内容可能有误,请谨慎听取。 TaskToDoList-master 1.XAML “XAML”是WPF中专门用于设计UI的语言,优点是 1.XAML可以设计出专业的UI和…...
【kali靶机之serial】--反序列化漏洞实操
kali靶机配置 【我图片里没有截图的默认配置即可】需要改的地方图片里面都有。 使用kali扫描网关的主机。 扫到一个开放了80端口HTTP协议的主机ip 访问80端口 会看到一个文本页面,翻译一下看是什么意思。。 F12查看cookie,是一个base64编码了的东西 使…...
学习大数据DAY34 面向对象思想深化练习 将从豆瓣爬取的数据置入自己搭建的网站上
目录 查看电影类型的电影列表 添加电影 修改电影 上机练习 13 使用三层架构完善 web 系统 查看电影类型的电影列表 DAL.py 文件 class MovieDAL(DBHelper): def getMovieByTid(self,typeid): sqlf"""select id,title,release_date,score,tname from Mo…...
【开端】通过Java 过滤器灵活配置URL访问权限,并返回403
一、绪论 在JAVA项目系统中,后端给前端提供接口。但是在某些场景我们需要临时控制接口是否能被访问。或关闭某一接口的访问权限。 比如某一接口被攻击了或者某一接口存在漏洞,在系统不关闭的情况下,如何控制系统的访问权限。 二、控制接口访…...
【C++综合项目】——基于Boost库的搜索引擎(手把手讲解,小白一看就会!!)
目录 一、前言 二、项目的相关背景 ⚡什么是Boost库?⚡ ⚡什么是搜索引擎?⚡ ⚡为什么要做Boost搜索引擎?⚡ 二、搜索引擎的宏观原理 三、搜索引擎技术栈和项目环境 四、正排索引 VS 倒排索引 —— 搜索引擎的具体原理 &#x…...
华为云AI开发平台ModelArts
华为云ModelArts:重塑AI开发流程的“智能引擎”与“创新加速器”! 在人工智能浪潮席卷全球的2025年,企业拥抱AI的意愿空前高涨,但技术门槛高、流程复杂、资源投入巨大的现实,却让许多创新构想止步于实验室。数据科学家…...
OpenLayers 可视化之热力图
注:当前使用的是 ol 5.3.0 版本,天地图使用的key请到天地图官网申请,并替换为自己的key 热力图(Heatmap)又叫热点图,是一种通过特殊高亮显示事物密度分布、变化趋势的数据可视化技术。采用颜色的深浅来显示…...
MFC内存泄露
1、泄露代码示例 void X::SetApplicationBtn() {CMFCRibbonApplicationButton* pBtn GetApplicationButton();// 获取 Ribbon Bar 指针// 创建自定义按钮CCustomRibbonAppButton* pCustomButton new CCustomRibbonAppButton();pCustomButton->SetImage(IDB_BITMAP_Jdp26)…...
线程与协程
1. 线程与协程 1.1. “函数调用级别”的切换、上下文切换 1. 函数调用级别的切换 “函数调用级别的切换”是指:像函数调用/返回一样轻量地完成任务切换。 举例说明: 当你在程序中写一个函数调用: funcA() 然后 funcA 执行完后返回&…...
将对透视变换后的图像使用Otsu进行阈值化,来分离黑色和白色像素。这句话中的Otsu是什么意思?
Otsu 是一种自动阈值化方法,用于将图像分割为前景和背景。它通过最小化图像的类内方差或等价地最大化类间方差来选择最佳阈值。这种方法特别适用于图像的二值化处理,能够自动确定一个阈值,将图像中的像素分为黑色和白色两类。 Otsu 方法的原…...
Keil 中设置 STM32 Flash 和 RAM 地址详解
文章目录 Keil 中设置 STM32 Flash 和 RAM 地址详解一、Flash 和 RAM 配置界面(Target 选项卡)1. IROM1(用于配置 Flash)2. IRAM1(用于配置 RAM)二、链接器设置界面(Linker 选项卡)1. 勾选“Use Memory Layout from Target Dialog”2. 查看链接器参数(如果没有勾选上面…...
Spring Boot面试题精选汇总
🤟致敬读者 🟩感谢阅读🟦笑口常开🟪生日快乐⬛早点睡觉 📘博主相关 🟧博主信息🟨博客首页🟫专栏推荐🟥活动信息 文章目录 Spring Boot面试题精选汇总⚙️ **一、核心概…...
ArcGIS Pro制作水平横向图例+多级标注
今天介绍下载ArcGIS Pro中如何设置水平横向图例。 之前我们介绍了ArcGIS的横向图例制作:ArcGIS横向、多列图例、顺序重排、符号居中、批量更改图例符号等等(ArcGIS出图图例8大技巧),那这次我们看看ArcGIS Pro如何更加快捷的操作。…...
Fabric V2.5 通用溯源系统——增加图片上传与下载功能
fabric-trace项目在发布一年后,部署量已突破1000次,为支持更多场景,现新增支持图片信息上链,本文对图片上传、下载功能代码进行梳理,包含智能合约、后端、前端部分。 一、智能合约修改 为了增加图片信息上链溯源,需要对底层数据结构进行修改,在此对智能合约中的农产品数…...
NXP S32K146 T-Box 携手 SD NAND(贴片式TF卡):驱动汽车智能革新的黄金组合
在汽车智能化的汹涌浪潮中,车辆不再仅仅是传统的交通工具,而是逐步演变为高度智能的移动终端。这一转变的核心支撑,来自于车内关键技术的深度融合与协同创新。车载远程信息处理盒(T-Box)方案:NXP S32K146 与…...
