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

C++模版初阶

泛型编程

如下的交换函数中,它们只有类型的不同,应该怎么实现一个通用的交换函数呢?

void Swap(int& left, int& right)
{int temp = left;left = right;right = temp;
}void Swap(double& left, double& right)
{double temp = left;left = right;right = temp;
}void Swap(char& left, char& right)
{char temp = left;left = right;right = temp;
}

使用函数重载虽然可以实现,但是有一下几个不好的地方:
        1. 重载的函数仅仅是类型不同,代码复用率比较低,只要有新类型出现时,就需要用户自己增加对应的函数;
        2. 代码的可维护性比较低,一个出错可能所有的重载均出错。

那能否告诉编译器一个模子,让编译器根据不同的类型利用该模子来生成代码呢? 

        如果在C++中,也能够存在这样一个模具,通过给这个模具中更换不同的字体(类型),来获得不同的模版(即生成具体类型的代码),那将会节省许多头发和时间。巧的是前人早已将树栽好,我们只需在此乘凉。有了如上图的思想,我们也可以创建这样的一个模版!就有了泛型编程。
泛型编程:编写与类型无关的通用代码,是代码复用的一种手段。模板是泛型编程的基础。

模版分为:

函数模版:

函数模版概念:

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

函数模板格式:

格式:template<typename Type>
           template<class Type>

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

函数模版原理:

        函数模板是一个蓝图,它本身并不是函数,是编译器用使用方式产生特定具体类型函数的模具。所以其实模板就是将本来应该我们做的重复的事情交给了编译器。

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

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

函数模版实例化: 

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

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

隐式实例化: 让编译器根据实参推演模板参数的实际类型

 显式实例化:在函数名后的<>中指定模板参数的实际类型

模版参数的匹配原则:

1. 一个非模板函数可以和一个同名的函数模板同时存在,而且该函数模板还可以被实例化为这个非模板函数;
2. 对于非模板函数和同名函数模板,如果其他条件都相同,在调动时会优先调用非模板函数而不会从该模板产生出一个实例。如果模板可以产生一个具有更好匹配的函数, 那么将选择模板;

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

 可以理解为:

类模版

类模版的定义格式:

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

 类模版实例化:

#include <iostream>
using namespace std;template<class T>
class Stack
{
public:Stack(int n = 4);~Stack(){cout << "~Stack()" << endl;delete[] _a;_a = nullptr;_top = _capacity = 0;}void Push(const T& x){//...}private:T* _a;int _top;int _capacity;
};//类模版不能将定义和声明在不同的文件
//类外定义
template<class T>
Stack<T>::Stack(int n)
{cout << "Stack(int n = 4)" << endl;_a = new T[n];_top = 0;_capacity = n;
}int main()
{//在普通的类中:类名就是类型//在类模版实例化的类中:类名不是类型,类名<数据类型>才是类型Stack<int> st1;//显示实例化Stack<double> st2;//显示实例化的类型不同,它们就是不同的类//st1 = st2;//一个是Stack<int>  一个是Stack<double>return 0;
}

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

以上就是个人学习的见解和学习过程的解析,欢迎各位大佬在评论区探讨,交流!
如果本篇对你有帮助的话,三连支持一下吧。
感谢大佬们的三连! 感谢大佬们的三连! 感谢大佬们的三连!

                                              

http://www.lryc.cn/news/234534.html

相关文章:

  • ​软考-高级-系统架构设计师教程(清华第2版)【第20章 系统架构设计师论文写作要点(P717~728)-思维导图】​
  • Go 语言结构体验证详解:validate 标签与自定义规则
  • ​软考-高级-系统架构设计师教程(清华第2版)【第19章 大数据架构设计理论与实践 (P691~716)-思维导图】​
  • 深度学习YOLOv5车辆颜色识别检测 - python opencv 计算机竞赛
  • c语言-浅谈指针(3)
  • 从服务器端获取人脸数据,在本地检测特征,并将特征发送给服务器
  • ARDUINO UNO 12颗LED超酷流水灯效果
  • Linux下查看pytorch运行时真正调用的cuda版本
  • ​分享mfc140u.dll丢失的解决方法,针对原因解决mfc140u.dll丢失的问题
  • torch_cluster、torch_scatter、torch_sparse三个包的安装
  • 软件安利——火绒安全
  • Induced AI:一个专门为自动化任务而设计的AI原生浏览器RPA平台
  • vue3中使用reactive定义的变量响应式丢失问题(大坑!!!)
  • Windows Server 2012 R2系统服务器远程桌面服务多用户登录配置分享
  • mysql之搭建MHA架构实现高可用
  • Databend 与海外某电信签约:共创海外电信数据仓库新纪元
  • scala解析命令行参数
  • 盘点60个Python各行各业管理系统源码Python爱好者不容错过
  • SpringSecurity6 | 自动配置(下)
  • 6、传统CV之均值滤波
  • 快速搭建本地的chatgpt
  • 分布式下多节点WebSocket消息收发
  • LeetCode算法题解(动态规划)|LeetCode509. 斐波那契数、LeetCode70. 爬楼梯、LeetCode746. 使用最小花费爬楼梯
  • 【图像处理】:Otsu算法最大类间方差法(大津算法:附源码)
  • 【uni-app】设置背景颜色相关
  • 工厂模式-C++实现
  • 安装应用与免安装应用差异对比
  • FiscoBcos使用Go调用合约
  • 自然语言处理(NLP)-spacy简介以及安装指南(语言库zh_core_web_sm)
  • CTF-PWN-tips