【C++进阶之路】模板

news/2024/12/21 22:20:07/

前言

假如需要你写一个交换函数,交换两个相同类型的值,这时如果交换的是int 类型的值,你可能会写一个Swap函数,其中参数是两个int类型的,假如再让你写一个double类型的呢?你可能又要写一个Swap的函数重载,参数是两个类型double的,假如再让你写一个char类型的呢?你可能……这样会有尽头吗?没有,因为还有自定义类型呢!那我们如何解决这样的问题呢?——模板(主要功能是实现通用)

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

一. 函数模板

基本用法

①定义模板参数类型名

template<typename T1, typename T2,......,typename Tn>
//这里的typename也可以换为class
template<class T1, class T2,......,class Tn>
  • 切记: 不能换为struct

②函数模板的实现

  • 比如我们实现一个交换函数的模板
template<typename T>
void Swap(T& n1, T& n2)
{T tmp = n1;n1 = n2;n2 = tmp;
}

用如下几个例子实验一下:
例1:

class Date
{
public:void Print(){cout << _year << "年" << _month << "月" << _day << "日" << endl;}Date(int year,int month,int day):_year(year),_month(month),_day(day){}
private:int _year;int _month;int _day;
};
int main()
{int x = 0,y = 1;double x1 = 1.1, y1 = 2.2;Date x2(1949, 10, 1), y2(2023, 5, 23);Swap(x, y);Swap(x1, y1);Swap(x2, y2);x2.Print();y2.Print();cout << "x:" << x << " y:" << y << endl;cout << "x1:" << x1 << " y1:" << y1 << endl;return 0;
}

运行结果:
在这里插入图片描述

  • 可以看到这是交换了。

到这里我们又有一个疑问——调用的是一个函数吗?
继续分析——转到反汇编
在这里插入图片描述

  • 可以看到——调用的Swap的地址是不一样的!
  • 因此:调用的不是一个函数。
  • 如何解释呢?

换汤不换药,换的是类型,不变的交换的思想。
图解:
在这里插入图片描述

其实我们也跟根本不用自己实现交换函数,直接用库里的就行了。在这里插入图片描述

  • 不过库里的是全小写,我们平常实现的是首字母大小。

③模板的实例化


隐式实例化

  • 编译器根据实参推演模板参数的实际类型
template <typename T>
T Func(const T& n1, const T& n2)
{return n1 + n2;
}
int main()
{int x = 0,y = 1;Func(x, y);return 0;
}
  • 就好比这里传参并没有明确的说明x的类型,但是编译器会自动推导实参的类型。

显示实例化

  • 在函数名后的<>中指定模板参数的实际类型
template <typename T>
T Func(const T& n1, const T& n2)
{return n1 + n2;
}int main()
{int x = 0,double y = 1;Func<int>(x, y);//这里会将y强制类型转换为一个double类型的常量,因此函数参数的const不可省去。return 0;
}
  • 像这样如果,编译器推导不出来类型,这时就需要我们显示声明一下,其中的类型是什么,否则编译器会直接报错。
  • 那除了显示示例化还有什么办法么?——强制类型转换。
template <typename T>
T Func(const T& n1, const T& n2)
{return n1 + n2;
}int main()
{int x = 0,double y = 1;Func(x, (int)y);//这里会将y强制类型转换为一个double类型的常量,因此函数参数的const不可省去。return 0;
}
  • 强转的结果必然是使两个参数类型保持一致,如果强转后的参数类型不一致,那么编译器还是会报错。

还有一种情况是我们必须显示示例化的——模板参数做返回值
示例:

template <typename T>
T Func(int x, int y)
{return x + y;
}int main()
{int x = 0;int y = 1;Func<int>(x,y);return 0;
}
  • 这就好比你要使用函数模板,得先让编译器知道或者能推导出模板参数的类型,那么返回值写上模板参数的话,编译器是无从下手的,那么只有显式实例化了,编译器才知道返回类型,才会给你生成指定类型的函数。

④函数模板与普通函数的调用规则

1.能用普通函数就用普通函数,用不了再用函数模板。

int add(int x, int y)
{return x + y;
}template <typename T1,typename T2>
int add(T1 x, T2 y)
{return x + y;
}int main()
{add(1, 2);//这里调用的非模板add(1.0, 1);//这里使用的是模板生成的函数return 0;
}

看反汇编:
在这里插入图片描述

2.非模板函数与模板部分重复都可以使用。

int add(int x, int y)
{return x + y;
}
template <typename T1,typename T2>
int add(T1 x, T2 y)
{return x + y;
}
int main()
{add(1, 2);//这里是调用add函数,因为能用普通函数就用普通函数add<int>(1, 1);这里是用的add模板函数实例化生成的函数。//这样也可以add<>(1, 1);//这算是空模板,只是为了调用模板而已。return 0;
}

3.普通函数支持类型转换,而模板要严格按照类型。

template<typename T>
int myAdd(T a, T b)
{cout << "template function" << endl;return a + b;
}int myAdd(char a, char b)
{cout << "normal function" << endl;return a + b;
}
void test()
{int a = 10;int b = 20;char c1 = 'a';char c2 = 'b';myAdd(a, c1);//这里调用的是函数,这里的a会自动转换为char类型的myAdd(a, b);//两个都是int类型的调用模板函数myAdd(c1, b);//这里会发生类型转换,b的int类型会转换为char类型的myAdd(c1, c2);myAdd<>(c1, c2);//这里是显示实例化,自然会调用函数模板
}
int main()
{test();return 0;
}
  • 总结:

1.普通函数优先调用
2.普通函数支持类型转换
3.模板必须严格的类型匹配
4.如果函数模板可以产生一个更好的匹配,那么选择模板。
5.可以用空模板或者模板实例化让编译器只调用模板

二. 类模板

  • 简而言之就是类是通用的,比如一个栈既要存int类型的,又要存double类型等等。
  • 说明:类模板不是具体的类,是用来生成类对象的模具。

  • 补充:template定义的模板参数只在最近的一个大括号内有效。
#include<iostream>
using namespace::std;
template <typename T>
class Stack
{
public:Stack(int capacity = 4){cout << "Stack()" << endl;T* tmp = new T[capacity];_arr = tmp;_top = 0;_capacity = capacity;}void PushBack(T x);private:T* _arr;int _top;int _capacity;
};
//这里放在类外进行定义。
template <typename T>
//这里要说明的是必须写Stack<T>这是具体的类,而不能只写Stack。
void Stack<T>::PushBack(T x)
{if (_top == _capacity){T* tmp = (T*)realloc(_arr, sizeof(T) * _capacity * 2);if (tmp == NULL){perror("realloc fail");exit(-1);}else{_arr = tmp;_capacity *= 2;}}_arr[_top++] = x;
}
int main()
{Stack<int> stack1;//这要说明使用的数据类型,也就是类模板示例化。return 0;
}// 注意:类模板
//1.类名——Stack
//2.类型 ——Stack<T>
  • 类模板实例化得到的类叫模板类

http://www.ppmy.cn/news/90689.html

相关文章

牛顿-莱布尼茨公式

前置知识&#xff1a;黎曼积分的概念 牛顿-莱布尼茨公式 设 f f f在 [ a , b ] [a,b] [a,b]上可积&#xff0c;令 F ( x ) ∫ a x f ( t ) d t F(x)\int_a^xf(t)dt F(x)∫ax​f(t)dt 则 &#xff08;1&#xff09; F F F在 [ a , b ] [a,b] [a,b]上连续 &#xff08;2&…

如何清理harbor的磁盘空间

博客主页&#xff1a;https://tomcat.blog.csdn.net 博主昵称&#xff1a;农民工老王 主要领域&#xff1a;Java、Linux、K8S 期待大家的关注&#x1f496;点赞&#x1f44d;收藏⭐留言&#x1f4ac; 目录 registry garbage-collectharbor自带的清理工具docker image prune -a…

基于SSM的校园办公管理系统的设计与实现(源码完整)

项目描述 临近学期结束&#xff0c;还是毕业设计&#xff0c;你还在做java程序网络编程&#xff0c;期末作业&#xff0c;老师的作业要求觉得大了吗?不知道毕业设计该怎么办?网页功能的数量是否太多?没有合适的类型或系统?等等。这里根据你想解决的问题&#xff0c;今天给…

ROS学习——利用电脑相机标定

一、 安装usb-cam包和标定数据包 sudo apt-get install ros-kinetic-usb-cam sudo apt-get install ros-kinetic-camera-calibration 要把kinetic改成你自己的ros版本 。 二、启动相机 roslaunch usb_cam usb_cam-test.launch 就会出现一个界面 可以通过下面命令查看相机…

Flutter 可冻结的侧滑表格 sticky-headers-table 结合 NestedScrollView 吸顶悬浮的使用实践

最近在做flutter web的开发&#xff0c;需要做一个类似云文档中表格固定顶部栏和左侧栏的需求&#xff0c;也就是冻结列表的功能 那么在pub上呢也有不少的开源库&#xff0c;比如&#xff1a; table_sticky_headers data_table_2 如果说只是简单的表格和吸顶&#xff0c;那么这…

vue3前台查询使用多个字典项并且和后台交互

目录 一、前端使用 1.前台vue3接口使用 dictManege.ts 2.前台使用该接口地方 3.前台反显地方 其他几个都一样&#xff0c;这里使用在state中定义的idTypeList,在上面赋值&#xff0c;在这里使用 二、后端使用 4.后端controller接口实现 其中使用字典String[]来接收 放…

对于大流量请求的处理方案(NATNginx)

情况描述&#xff1a; 如图所示&#xff0c;厂家的A服务器&#xff0c;到客户的C服务器不通&#xff0c;需要我这边通过B服务器做一次流量转发。 由于&#xff0c;每次请求数据流都太大&#xff0c;怕HTTPS方式&#xff0c;会出现请求超时&#xff0c;断开连接。 解决方案&am…

目标检测数据预处理——非宫格与宫格混合拼图(大宽高比图片)

之前一直用的是宫格的正方形拼图&#xff0c;但比如对“人”框的截图是这种高宽高比的长方形图片&#xff0c;按照最大边resize最小边等比例缩放后放入宫格中对造成最小边resize太多&#xff0c;整体图片缩小很多。所以本片专门针对高宽高比的图片拼图进行编辑。 本篇的拼图方式…