LoadMap9:C++的Template模板函数

1. 模板函数与模板类

模板Template和函数重载是实现静态多态的两种重要途径。对于模板而言,其通常用于模板函数和模板类中。其基本语法结构为:

template <typename XXXX>  // XXXX 代表定义的模板数据类型的名称

1.1 模板函数

使用 template 关键字定义类型,用定义好的新类型定义函数类型以及其参数类型。调用模板函数式方法:函数名<函数参数类型>(参数列表),若实参并非该指定类型,则先进行强制转换,或者不显式说明函数参数类型,由编译器自主推断,此时输出参数的类型必须一致。

#include<iostream>
using namespace std;// 1.1 模板函数
template<typename ftype>            // 使用 template 关键字定义类型, 可以同时定义多个类型
ftype myMax(ftype a, ftype b){      // 用定义好的新类型定义函数类型以及其参数类型return (a > b) ? a : b;
}int main(){int a = 10, b = 15;int ans1 = myMax<int>(a, b);        // 调用模板函数方法1:函数名<函数参数类型>(参数列表)int ans2 = myMax(a, b);             // 调用目标函数方法2:不显式说明函数参数类型,由编译器自主推断cout << "ans1: " << ans1 << endl;cout << "ans2: " << ans2 << endl;
}

1.2 模板类

类似于模板类,使用 template 关键字定义类型,可以同时定义多个类型,然后就可以用定义好的新类型定义成员变量的函数类型,类中构造函数/其他函数使用模板参数类型就类似于模板函数的使用方法。调用方法可以不显式说明函数参数类型,由编译器自主推断,也可以函数名<函数参数类型>(参数列表),若实参并非该指定类型,则先进行强制转换。

#include<iostream>
using namespace std;// 1.2 模板类
template <typename ctype, typename btype>       // 使用 template 关键字定义类型,可以同时定义多个类型
class myclass{public:ctype m_var1;           // 用定义好的新类型定义成员变量的函数类型btype m_var2;myclass(ctype var1, btype var2) : m_var1(var1), m_var2(var2){   // 构造函数/其他函数:类似于模板函数的使用方法cout << "m_var1: " << m_var1 << endl;cout << "m_var2: " << m_var2 << endl;};
};int main(){int a = 10;float b = 15;myclass cls1(a, b);             // 调用方法1:不显式说明函数参数类型,由编译器自主推断myclass<int, float> cls2(a, b);     // 调用模板函数方法2:函数名<函数参数类型>(参数列表),若实参并非该指定类型,则先进行强制转换
}

1.3 模板特定化

类似于函数重载,当我的同一个名字的类有多种不同的实现方法时,我希望编译器能够根据我指定的类参数类型自动匹配对应的类实现方法时,可以使用模板特定化的功能。以下展示 myclass 类的两种使用方法,当指定类型为 char 类型时,则编译器自动调用第二种类型,即把小写字母转化为大写字母,否则自动调用第一种类的实现方法,即直接打印输出。

#include<iostream>
using namespace std;// myclass 模板类的第一种实现方法
template <typename ctype, typename btype>       // 使用 template 关键字定义类型,可以同时定义多个类型
class myclass{ public:ctype m_var1;           // 用定义好的新类型定义成员变量的函数类型btype m_var2;myclass(ctype var1, btype var2) : m_var1(var1), m_var2(var2){   // 构造函数/其他函数:类似于模板函数的使用方法cout << "m_var1: " << m_var1 << endl;cout << "m_var2: " << m_var2 << endl;};
};// myclass 模板类的第二种实现方法
template <>
class myclass<char, char> {       // 类型为 char 型时调用
public:char first;char second;myclass(char first, char second) : first(first), second(second) {// Special behavior for characters (e.g., convert to uppercase)cout << char(std::toupper(this->first)) << endl;   // 转化为大写cout << char(std::toupper(this->second)) << endl;};
};int main(){int a = 10;float b = 15;char c = 'c', d = 'd';myclass<int, float> cls2(a, b);     // 调用 myclass 模板类的第一种实现方法, 输出 m_var1: 10, m_var2: 15myclass cls1(c, d);             // 调用 myclass 模板类的第二种实现方法,输出 C D
}

以上例子的第二种 myclass 的实现方法展示了如何使用全部模板特定化,可以看到,其基础的语法格式为:

// 模板特定化定义方法
template<>   // <> 内不设置任何的参数
class ClassName <基础类型1,基础类型2>{}

当然,除了全部模板特定化,还可以只定义所有模板种的部分为特定的基础类型,编译器也会根据部分匹配来优先选择合适的模板函数。其基础语法如下:

// 部分模板特定化定义方法
// 例子1:部分参数特定化
template<typename T>   // <> 内设置自定义的参数类型
class ClassName <T, char>{};  	// 带有 char 类型的变量自动调用该类 
// 例子2:部分指针特定化
template<typename T>   // <> 内设置自定义的参数类型
class ClassName <T*>{};  // 指针变量则自动调用模板类

2. 可变参数模板

首先什么场景需要用到可变参数模板呢?如1.2中的例子,我需要初始化两个参数,所以,我们模板函数的写法为:

template <typename ctype, typename btype>       // 使用 template 关键字定义类型,可以同时定义多个类型

当我需要初始50个或者未知个数的参数时,这种方法就显得非常鸡肋了,因此引入可变参数模板,其基础语法为:

template <typename... Args>   // 其中 Args 代表参数类型的列表

因为不确定模板参数类型的个数有多少,因此常常通过递归调用的方法实现。以下通过一个函数和一个类展示其用法。

#include<iostream>
using namespace std;// 2.1 可变参数模板函数
template <typename T>   // base case:当递归到最后一个元素时会调用该函数
T mysum(T t) {return t;
}template <typename T, typename... ARGS>     // 可以适用于未知个数函数参数
T mysum(T t, ARGS... args){return t + mysum(args...);
}// 2.2 可变参数模板类template <typename... Types>
class myclass;// 类似于函数,也需要额外定义 base case,即递归到空类
template <>
class myclass<>{};template<typename Head, typename... Tail>
class myclass<Head, Tail...> : public myclass<Tail...>{public:myclass(Head head, Tail... tail) : myclass<Tail...>(tail...), head_(head){cout << head_ << endl;};private:Head head_;
};int main(){cout << mysum(1, 2, 3, 4, 5, 6) << endl;  // output: 21myclass<int, float, double, int> cls(1, 2.0f, 3.0, 4);  // 当不显式设置参数类型时报错,无法自动推断,不解, output: 4, 3, 2, 1
}

3. 类型特征与模板函数

函数特征,通过头文件 #include<type_traits> 添加。其核心功能在判断输入变量是否属于某一个特定的类型,如:

#include<iostream>
#include<type_traits>
using namespace std;int main(){int a = 10;cout << "is it a pointer? " << boolalpha << is_pointer<decltype(a)>::value <<endl;     // is it a pointer? falsecout << "is it a arithmetic? " << boolalpha << is_arithmetic<decltype(a)>::value <<endl;  // is it a arithmetic? truecout << "is it a function? " << boolalpha << is_function<decltype(a)>::value <<endl;  // is it a function? false
}

那么这如何跟模板函数梦幻联动呢?通过 条件控制!

  1. std::conditional<condition, A, B> :如果条件为真,则使用模板类型A,否则使用类型B
  2. std::enable_if<condition, A>:如果条件为真,则使用模板类型A,否则没有输入类型
#include<iostream>
#include<type_traits>
using namespace std;template <typename T>
typename enable_if<is_arithmetic<T>::value, T>::type find_max(T a, T b){cout << "max value is: " << (a > b ? a : b) << endl;
};template <typename T>
typename conditional<is_arithmetic<T>::value, T, char>::type find_max1(T a, T b){cout << "max value is: " << (a > b ? a : b) << endl;
}int main(){int a = 10, b = 25;find_max(a, b);find_max1(a, b);
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

User-Agent(用户代理)是什么?

User-Agent&#xff08;用户代理&#xff09;是什么&#xff1f; User-Agent 即用户代理&#xff0c;简称“UA”&#xff0c;它是一个特殊字符串头。网站服务器通过识别 “UA”来确定用户所使用的操作系统版本、CPU 类型、浏览器版本等信息。而网站服务器则通过判断 UA 来给客…

Android 13 关闭相册的编辑功能

介绍 因为做的是学生机&#xff0c;客户不希望相册的图片可以编辑。 分析 通过字符串我们找到了几个资源文件&#xff0c;以下只展示其中一个 路径&#xff1a;vendor/mediatek/proprietary/packages/apps/Gallery2/res/menu/operation.xml <item android:id"id/acti…

python3.7conda安装opencv

1.conda创建虚拟环境 conda create --name opencv3.7 python3.72.激活虚拟环境 conda activate opencv3.7 3.安装 opencv pip install -i https://pypi.mirrors.ustc.edu.cn/simple/ opencv-python3.4.2.16 python3.7安装opencv python 和 opencv-contrib-python的相关问题…

Hotspot源码解析-第十九章-ClassLoaderData、符号表、字符串表的初始化

第十九章-ClassLoaderData初始化 讲解本章先从一张图开始 众所周知&#xff0c;Java类的相关信息都是存储在元空间中的&#xff0c;但是是怎么存储的&#xff0c;相信很多读者是不清楚的&#xff0c;这里就不得不涉及到ClassLoaderDataGraph、classLoader、classLoaderData&…

MIT 6s081 lab 5: xv6 lazy page allocation

Page faults Basic 通过page fault可以实现一系列的虚拟内存功能&#xff1a; lazy allocationcopy-on-write forkdemand pagingmemory mapped files 虚拟内存的两个主要的优点&#xff1a; 1、隔离性&#xff1a;每个应用程序拥有自己的地址空间&#xff0c;因此不可能修…

【Dart】=> [06] Dart初体验-类Class-构造函数-继承-mixin-异步编程-链式调用-泛型-异常

目录 能够定义并使用Dart的类类的定义构造函数私有属性和方法继承mixin异步编程FutureFuture链式调用async - awaitdynamic类型泛型异常 能够定义并使用Dart的类 Dart是一门面向对象的编程语言&#xff0c;所有的对象都是类的实例 通过类我们可以对数据和方法进行封装复用 学习…

(2023版)斯坦福CS231n学习笔记:DL与CV教程 (1) | 引言与知识基础

前言 &#x1f4da; 笔记专栏&#xff1a;斯坦福CS231N&#xff1a;面向视觉识别的卷积神经网络&#xff08;23&#xff09;&#x1f517; 课程链接&#xff1a;https://www.bilibili.com/video/BV1xV411R7i5&#x1f4bb; CS231n: 深度学习计算机视觉&#xff08;2017&#xf…

NLP论文阅读记录 - 2022 | WOS 用于摘要法律文本的有效深度学习方法

文章目录 前言0、论文摘要一、Introduction1.1目标问题 二.相关工作三.本文方法四 实验效果4.1数据集4.2 对比模型4.3实施细节4.4评估指标4.5 实验结果4.6 细粒度分析 五 总结 前言 Effective deep learning approaches for summarization of legal texts&#xff08;22&#x…

Linux -- Nginx服务基础

4.1Nginx服务基础 Nginx(发音为[engine x])专为性能优化而开发&#xff0c;其最知名的优点是它的稳定性和低系统资源消 耗&#xff0c;以及对HTTP并发连接的高处理能力&#xff08;单台物理服务器可支持30000~50000个并发请求&#xff09;&#xff0c;正因 为如此&#xff0c;…

html中flex的使用

在HTML中&#xff0c;flex属性用于设置弹性容器的子元素的布局方式。使用flex属性可以实现灵活的布局&#xff0c;使子元素根据可用空间自动调整大小。flex属性有三个值&#xff1a;flex-grow、flex-shrink和flex-basis。 flex-grow&#xff1a;指定子元素的放大比例&#xff…

路由器路由配置解析

路由器是网络中负责转发数据包的设备&#xff0c;通过配置路由规则&#xff0c;确定数据包的传输路径。在本文中&#xff0c;我们将解析一个路由器的配置&#xff0c;并说明每个路由规则的含义。 路由器配置 rootr-63-VM:# ip route default via 192.168.157.2 dev eth1 10.1…

vscode中关于python的一些常用配置

文章目录 python cv2 提示配置第一步 配置提示信息第二部 重启vs 可能还不行&#xff0c;那就重新安装以下opencv-python 配置pytest还是如上&#xff0c;将下入的位置打开编写测试用例 配置跨文件import在工作目录中新建一个.env文件输入内容如下打开.vscode中的setting.json …

目标识别跟踪模块Tofu3

Tofu系列提供了适应不同目标、不同速率的识别跟踪模块产品系列&#xff0c;主要包括Tofu3&#xff0c;4&#xff0c;5&#xff0c;S和其他零配件&#xff0c;可以适配BT.656,Cameralink&#xff0c;网络等不同接口和协议的热红外、可见光视频。 Tofu3 是多波段视频物体识别跟踪…

PTA 7-27 输出下半张九九乘法表

请输出下半张九九乘法表&#xff0c;即下三角的半张。 11 1 21 2 22 4 31 3 32 6 33 9 41 4 42 8 4312 4416 51 5 5210 5315 5420 5525 61 6 6212 6318 6424 6530 6636 71 7 7214 7321 7428 7535 7642 7749 81 8 8216 8324 8432 8540 8648 8756 8864 91 9 9218 9327 9…

GL Logger和CANFDLog-OTL-128两款记录仪都是如何实现高效的报文录制的?

GL Logger是Vector推出的记录CAN/CAN FD、LIN、FlexRay和MOST数据通信的工具。以GL2400为例带着大家一步步地实现路试过程中通过整车OBD口进行CAN/CANFD报文的录制。 Step1 设备配置 设备配置即设备录制方式、录制内容、设备休眠唤醒策略等。 ▷ 打开Vector Logger Configurat…

AMEYA360:帝奥微车规级高性能电平转换器 — DIA7B104

电平转换器(Level Shifter&#xff0c;LS)是一个在SOC设计中经常会用到的器件。它的主要作用是将数字信号从一个电压域切换到另一个电压域。随着汽车电气化和智能化的发展&#xff0c;汽车电子系统越来越复杂&#xff0c;各种功能模块之间的通讯也越来越多。由先进工艺制造的主…

UML-用例图

提示&#xff1a;用例图是软件建模的开始&#xff0c;软件建模中的其他图形都将以用例图为依据。用例图列举了系统所需要实现的所有功能&#xff0c;除了用于软件开发的需求分析阶段&#xff0c;也可用于软件的系统测试阶段。 UML-用例图 一、用例图的基础知识1.用例图的构成元…

openssl3.2 - 官方demo学习 - mac - hmac-sha512.c

文章目录 openssl3.2 - 官方demo学习 - mac - hmac-sha512.c概述笔记END openssl3.2 - 官方demo学习 - mac - hmac-sha512.c 概述 MAC算法为HMAC, 设置参数(摘要算法为SHA3-512), 用key初始化, 对明文做MAC数据. 笔记 /*! \file hmac-sha512.c \note openssl3.2 - 官方demo…

使用numpy处理图片——滤镜

大纲 3维数组切分打平重组法深度切分法 3维数组堆叠 我们在用手机拍照片时&#xff0c;往往会对照片进行滤镜处理&#xff0c;从而让照片更加美观。本文我们将实现几种滤镜效果——去除所有像素中的某一种原色&#xff0c;形成只有红绿、红蓝和绿蓝原色的照片。 为了突出色彩丰…

Leetcode面试经典150题刷题记录 —— 数学篇

Leetcode面试经典150题刷题记录-系列Leetcod面试经典150题刷题记录——数组 / 字符串篇Leetcod面试经典150题刷题记录 —— 双指针篇Leetcod面试经典150题刷题记录 —— 矩阵篇Leetcod面试经典150题刷题记录 —— 滑动窗口篇Leetcod面试经典150题刷题记录 —— 哈希表篇Leetcod…