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

【C++】初识C++模板与STL

在这里插入图片描述

C++语法相关知识点可以通过点击以下链接进行学习一起加油!
命名空间缺省参数与函数重载C++相关特性类和对象-上篇类和对象-中篇
类和对象-下篇日期类C/C++内存管理

本章将简单分享C++模板与STL相关知识,与之相关更多知识将留到下次更详细地来分享给大家

请添加图片描述
Alt
🌈个人主页:是店小二呀
🌈C语言笔记专栏:C语言笔记
🌈C++笔记专栏: C++笔记
🌈初阶数据结构笔记专栏: 初阶数据结构笔记
🌈Linux笔记专栏: Linux笔记

🌈喜欢的诗句:无人扶我青云志 我自踏雪至山巅 请添加图片描述

文章目录

  • 一、泛型编程
  • 二、函数模板
    • 2.1 函数模板原理
    • 2.2 函数模板实例化
    • 2.3 模板参数匹配原则
  • 三、类模板
    • 3.1 类模板定义格式
    • 3.2 类模板的实现化
  • 四、简单了解STL
    • 4.1 STL的版本
    • 4.2 STL的六大组件
    • 4.3 如何学习STL
    • 4.4 STL的缺陷:

一、泛型编程

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

具体场景:实现一个通用的交换函数

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;
}
.....................

通过函数重载实现通用函数缺陷:

  • 重载的函数仅仅是类型不同,代码复用率比较低,只要有新类型出现时,就需要用户自己增加对应的函数
  • 代码的可维护性比较低,一个出错可能所有的重载均出错

在这里插入图片描述

如果在C++中,存在一个摸具,通过给这个摸具中填充不同材料(类型),赖获得不同材料的锻件(即生成具体类型的代码),那么就会节省许多头发。对此C++提出模板的概念,对于模板分为函数模板以及类模板。

在这里插入图片描述

二、函数模板

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

函数模板格式:template<typename T1,typename T2,.....,tyename Tn>

使用函数模板实现通用交换函数

template<typename T>
void Swap(T& left, T& right)
{T temp = left;left = right;right = temp;
}
int main()
{int a = 10; int b = 20;cout << a << "/" << b<<endl;Swap(a, b);cout << a << "/" << b << endl;double c = 1.1; double d = 2.2;cout << c << "/" << d << endl;Swap(c, d);cout << c << "/" << d << endl;return 0;
}

在这里插入图片描述

其中支持typenameclass来定义模板参数关键字,但是不能使用struct来代替class定义。

提出思考:当我们编写了函数模板,两次函数调用是否为同一函数呢?

:调用不是同一个函数,虽然在调试中都执行到模板函数体中,但是这只是编译器为了方便观察进行的调整。对于不同类型所占用空间大小不是相同以及浮点数存储和释放都有自己的规定。(Swap函数在库实现好了并且C++有模块的概念,可以直接调用库中Swap函数)

2.1 函数模板原理

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

在这里插入图片描述

在编译器编译阶段,对于模板函数的使用,**编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用,函数传的是对象,模板传的是类型。**比如:当用double类型使用函数模板事,编译器通过对实参类型的推演,将T确定为double类型,然后产生一份专门处理double类型的代码,对于字符类型也是如此。

2.2 函数模板实例化

用不同类型的参数使用函数模板时,称为函数模板的实例化。模板参数实例化分为:隐式实例化和显式实例。

在这里插入图片描述

如图该语句不能通过编译器,由于编译期间,编译器进行实例化需要推演其实参类型。

报错理由:通过实参a1将T推演为int,通过实参d1将T推演为double类型,但是模板参数列表中只有一个T,编译器无法确定此处到底该将T确定为int或者double类型而报错。(在模板中编译器一般不会进行类型转换操作,因为一旦转化出问题,编译器就要背黑锅了。)

那么针对上面出现的问题,有三种解决方案

  • 用户自己来强制转化
  • 使用显式实例化
  • 添加一个T2

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

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

第一种:强制转化

int main()
{int a1 = 10,a2 = 20;double d1 = 10.2, d2 = 20.2;Add(a1, (int)d1);Add((double)a1, d2);return 0;
}

在这里插入图片描述

第二种:显式实例化

int main()
{int a1 = 10,a2 = 20;double d1 = 10.2, d2 = 20.2;Add<int>(a1, d1);Add<double>(a1, d1);	return 0;
}

如果类型不匹配,编译器会尝试进行隐式类型转换,如果无法转换成功编译器将会报错。

第三种:添加一个T2(这里返回值的类型,需要用户选择T1 or T2)

template<class T1,class T2>
T1 Add(const T1& left, const T2& right)
{cout << left + right<< endl;return left + right;
}int main()
{int a1 = 10,a2 = 20;double d1 = 10.2, d2 = 20.2;Add(a1, d1);Add(a1, d1);return 0;
}

2.3 模板参数匹配原则

当同名函数模板与非模板函数同时存在,该函数模板可以实例化为非函数模板。

// 专门处理int的加法函数
int Add(int left, int right)
{return left + right;
}
// 通用加法函数
template<class T>
T Add(T left, T right)
{return left + right;
}
void Test()
{Add(1, 2); // 与非模板函数匹配,编译器不需要特化Add<int>(1, 2); // 调用编译器特化的Add版本
}

如果同名函数模板与非函数模板,并且其他条件相同,在调用时会优先调用非函数模板(有现成的吃现成的菜)而不会从该模板实例化一个,除非模板可以产生一个具有更好匹配的函数,在调用时优先选择模板(现成的不好吃,不如吃自己做的)就像是想我委屈嫁给你,不如我找个有钱的大爷~

templace<class T1, class T2>
T1 Add(T1 left, T2 right)
{return left + right;
}
int main()
{int ret = Add(1,2.0);return 0;
}

在这里插入图片描述

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

三、类模板

3.1 类模板定义格式

template<class T1,class T2,....,class Tn>class 类模板名{//类内成员定义};
// 动态顺序表
// 注意:Vector不是具体的类,是编译器根据被实例化的类型生成具体类的模具
template<class T>
class Vector
{
public :Vector(size_t capacity = 10): _pData(new T[capacity]), _size(0), _capacity(capacity){}
// 使用析构函数演示:在类中声明,在类外定义。~Vector();void PushBack(const T& data)void PopBack()// ...
size_t Size() {return _size;}
T& operator[](size_t pos)
{assert(pos < _size);return _pData[pos];
}
private:T* _pData;size_t _size;size_t _capacity;
};
// 注意:类模板中函数放在类外进行定义时,需要加模板参数列表
template <class T>
Vector<T>::~Vector()
{if(_pData)delete[] _pData;_size = _capacity = 0;
}
int main()
{//显示模板实例化Vector<int> d1(10);vector<double> d2(10.0);
}

类模板可以处理需要很多类型的数据,如果按照C语言那一套TypeData typename使用,当创建不同类型数据,需要修改名字连同实现逻辑也需要更换名字。不如使用模板,将我们需要重复做的事情交给编译器来做。

3.2 类模板的实现化

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

//Vector类名,Vector<int>才是类型
Vector<int> s1;
Vector<double> s2;

四、简单了解STL

STL(standard tmplate libaray-标准库):是C++标准库的重要组成部分,不仅是一个可复用的组件库,而且是一个包罗数据结构与算法的软件框架。

4.1 STL的版本

原始版本

Alexander Stepanov、Meng Lee 在惠普实验室完成的原始版本,本着开源精神,他们声明允许任何人任意运用、拷贝、修改、传播、商业使用这些代码,无需付费。唯一的条件就是也需要向原始版本一样做开源使
用。 HP 版本–所有STL实现版本的始祖。

P. J.版本:
由P. J. Plauger开发,继承自HP版本,被Windows Visual C++采用,不能公开或修改,缺陷:可读性比较低,符号命名比较怪异。

RW 版本:
由Rouge Wage公司开发,继承自HP版本,被C+ + Builder 采用,不能公开或修改,可读性一般。

SGI版本:
由Silicon Graphics Computer Systems,Inc公司开发,继承自HP版 本。被GCC(Linux)采用,可移植性好,可公开、修改甚至贩卖,从命名风格和编程 风格上看,阅读性非常高。我们后面学习STL要阅读部分源代码,
主要参考的就是这个版本

4.2 STL的六大组件

在这里插入图片描述

4.3 如何学习STL

在这里插入图片描述

4.4 STL的缺陷:

  1. STL库的更新太慢了。这个得严重吐槽,上一版靠谱是C++98,中间的C++03基本一些修订。C++11出来已经相隔了13年,STL才进一步更新
  2. STL现在都没有支持线程安全。并发环境下需要我们自己加锁。且锁的粒度是比较大的。
  3. STL极度的追求效率,导致内部比较复杂。比如类型萃取,迭代器萃取
  4. STL的使用会有代码膨胀的问题,比如使用vector/vector/vector这样会生成多份代码,当然这是模板语法本身导致的
  5. 在不久的将来,将更加深入的学习模板的进阶知识。

以上就是本篇文章的所有内容,在此感谢大家的观看!这里是店小二呀C++笔记,希望对你在学习C++语言旅途中有所帮助!
请添加图片描述

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 景芯SoC A72实战反馈
  • 冲刺蓝桥杯第四章标准模版库STL(上)
  • App推广新姿势:Xinstall地推码,一键直达用户心!
  • 嵌入式高频面试题——第一章 进程与线程(上)
  • Redis Stream 助力:打造实时用户行为日志处理平台
  • 分类预测|基于麻雀优化核极限学习机的数据分类预测Matlab程序SSA-KELM 多特征输入多类别输出 含基础KELM
  • BlinqIO:业界首个生成式 AI 测试平台
  • OpenCV绘图函数(5)绘制标记函数drawMarker()的使用
  • 【Cadence24】如何给PCB板露铜处理
  • pycharm修改文件大小限制
  • Docker网络模型深度解析
  • Java方法的引用
  • Linux —— 驱动——platform平台总线
  • 浅析 Linux 进程地址空间
  • C# 使用 StackExchange nuget 包进行 Redis操作
  • SegmentFault for Android 3.0 发布
  • 【干货分享】SpringCloud微服务架构分布式组件如何共享session对象
  • 0x05 Python数据分析,Anaconda八斩刀
  • Android开发 - 掌握ConstraintLayout(四)创建基本约束
  • JavaScript服务器推送技术之 WebSocket
  • Java-详解HashMap
  • JSONP原理
  • Linux后台研发超实用命令总结
  • node入门
  • Spring声明式事务管理之一:五大属性分析
  • VirtualBox 安装过程中出现 Running VMs found 错误的解决过程
  • Vue 重置组件到初始状态
  • Vue源码解析(二)Vue的双向绑定讲解及实现
  • 关于 Linux 进程的 UID、EUID、GID 和 EGID
  • 规范化安全开发 KOA 手脚架
  • 使用 Docker 部署 Spring Boot项目
  • 怎样选择前端框架
  • 正则学习笔记
  • 3月7日云栖精选夜读 | RSA 2019安全大会:企业资产管理成行业新风向标,云上安全占绝对优势 ...
  • mysql 慢查询分析工具:pt-query-digest 在mac 上的安装使用 ...
  • 国内唯一,阿里云入选全球区块链云服务报告,领先AWS、Google ...
  • ​ssh-keyscan命令--Linux命令应用大词典729个命令解读
  • #if #elif #endif
  • #Linux(Source Insight安装及工程建立)
  • #stm32整理(一)flash读写
  • #我与Java虚拟机的故事#连载15:完整阅读的第一本技术书籍
  • (¥1011)-(一千零一拾一元整)输出
  • (1综述)从零开始的嵌入式图像图像处理(PI+QT+OpenCV)实战演练
  • (2024,LoRA,全量微调,低秩,强正则化,缓解遗忘,多样性)LoRA 学习更少,遗忘更少
  • (js)循环条件满足时终止循环
  • (libusb) usb口自动刷新
  • (补)B+树一些思想
  • (附源码)springboot猪场管理系统 毕业设计 160901
  • (力扣)1314.矩阵区域和
  • (每日一问)操作系统:常见的 Linux 指令详解
  • (四)c52学习之旅-流水LED灯
  • (转)Oracle存储过程编写经验和优化措施
  • (轉貼)《OOD启思录》:61条面向对象设计的经验原则 (OO)
  • .bat批处理(八):各种形式的变量%0、%i、%%i、var、%var%、!var!的含义和区别
  • .net core 外观者设计模式 实现,多种支付选择