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

devtools/2024/11/15 0:47:44/

在这里插入图片描述

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++语言旅途中有所帮助!
请添加图片描述


http://www.ppmy.cn/devtools/100415.html

相关文章

CSS的简单介绍

1.什么是CSS CSS(层叠样式表)&#xff0c;用于控制页面的样式&#xff0c;简单地来说&#xff0c;CSS就是用来美化页面的一种语言。 2.基本语法规范 CSS的基本语法规范&#xff1a;CSS选择器{1或多条声明} 其中CSS选择器决定找谁(针对哪个元素进行修改)&#xff0c;声明决定…

Debian Linux上安装Jumpserver

1.安装 Debian并配置 登录www.debian.io,下载网络版安装&#xff0c;安装很快&#xff0c;但完成后修改IP就遇到问题vi /etc/network/interfaces auto eth0 #设置开机自动连接网络 iface lo inet loopback allow-hotplug eth0 iface eth0 inet static #static表示使用固定I…

HTML 全解析:从基础到实战

一、简介 HTML&#xff08;HyperText Markup Language&#xff09;即超文本标记语言&#xff0c;是用于创建网页的标准标记语言。它通过各种标签来定义网页的结构和内容&#xff0c;使得浏览器能够正确地显示网页。HTML 文档由 HTML 元素组成&#xff0c;这些元素通过标签来表…

选择一家正规的应急指挥中心控制台厂家有多关键

在当今社会&#xff0c;随着自然灾害、突发事件及安全挑战的日益复杂多变&#xff0c;应急管理体系的构建显得尤为重要。而应急指挥中心作为应对各类紧急情况的神经中枢&#xff0c;其高效运作离不开先进、可靠的控制台设备支持。在此背景下&#xff0c;选择一家正规的应急指挥…

湖南贝特新能源科技:巧用草料二维码,实现设备管理数字化

在当今快速发展的制造业环境中&#xff0c;设备管理效率直接影响着企业的生产力与竞争力&#xff0c;我司也面临着设备管理流程中的诸多挑战。 如今购物买菜都是扫码支付的时代&#xff0c;我司也与时俱进&#xff0c; 导入了二维码系统进行公司质量、设备、安全、项目、改善等…

python如何快速生成一个密钥

在Python中&#xff0c;快速生成一个密钥通常依赖于内置的库或第三方库来生成一个安全的随机字符串。以下是一些常见的方法来生成密钥&#xff1a; 使用secrets模块 Python 3.6及以上版本引入了secrets模块&#xff0c;它用于生成适合管理密码、账户认证信息、安全令牌等敏感…

【JavaEE】SpringBoot 统一功能处理:拦截器、统一数据返回与异常处理的综合应用与源码解析

目录 SpringBoot 统⼀功能处理拦截器拦截器快速⼊⻔拦截器详解拦截路径拦截器执⾏流程 登录校验定义拦截器注册配置拦截器 DispatcherServlet 源码分析(了解)初始化(了解) DispatcherServlet的初始化1. HttpServletBean.init()2. FrameworkServlet.initServletBean() WebApplic…

DB-GPT开源项目论文解读

DB-GPT开源项目论文解读 (qq.com) 三篇文章的内容分别是1提出了一种开源的智能数据库对话系统、系统化方法、训练和推理策略&#xff1b;2提出了一个开源的Python库DB-GPT&#xff1b;3对各种开源大模型进行了系统的Text-to-SQL基准测试 DB-GPT&#xff1a;通过私有大型语言模…