【hello C++】模板初阶

news2024/12/24 8:23:37

目录

1. 泛型编程

2. 函数模板

2.1 函数模板的概念

2.2 函数模板格式

2.3 函数模板的原理

2.4 函数模板的实例化

2.5 模板参数的匹配原则

3. 类模板

3.1 类模板的定义格式

3.2 类模板的实例化

4. STL简介

4.1 什么是STL

4.2 STL的版本

4.3 STL的六大组件

4.4  STL的重要性

4.5 如何学习STL

4.6 STL的缺陷



 C++🌷

1. 泛型编程

下面请看一段代码:

#define _CRT_SECURE_NO_WARNINGS 1

#include <iostream>
using namespace std;

//整形数据交换
void Swap(int& x, int& y)
{
	int tmp = x;
	x = y;
	y = tmp;
}
//浮点型数据的交换
void Swap(float& x, float& y)
{
	float tmp = x;
	x = y;
	y = tmp;
}
//字符型数据
void Swap(char& x, char& y)
{
	char tmp = x;
	x = y;
	y = tmp;
}

int main()
{
	int a1 = 1, b1 = 2;
	Swap(a1, b1);
	float a2 = 1.2, b2 = 2.2;
	Swap(a2, b2);
	char a3 = 'x', b3 = 'y';
	Swap(a3, b3);
	cout << "a1:" << a1 << '\t' << "b1:" << b1 << endl;
	cout << "a2:" << a2 << '\t' << "b2:" << b2 << endl;
	cout << "a3:" << a3 << '\t' << "b3:" << b3 << endl;
	return 0;
}

在上述代码中使用函数重载,写了整型、浮点型、字符型的交换函数,完成了3组不同类型的数据

交换功能。

但看看代码,我们很容易发现它有以下几个不好的地方:

1. 重载的函数仅仅是类型不同,代码复用率比较低,只要有新类型出现时,就需要用户自己
增加对应的函数;
2. 代码的可维护性比较低,一个出错可能所有的重载均出错
那能否 告诉编译器一个模子,让编译器根据不同的类型利用该模子来生成代码 呢?

就如同上述的果冻膜具一样,我们往里面放什么佐料,做出来就是什么口味的果冻。

在C++中,也存在这样一个模具,通过给这个模具中填充不同材料(类型),来获得不同材料的

铸件(即生成具体类型的代码),这种编码方式便称为泛型编程

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

模板又分为:函数模板和类模板。

2. 函数模板

2.1 函数模板的概念

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

2.2 函数模板格式

template<typename T1,typename T2,...,typename Tn>
返回值类型 函数名(参数列表){}

typename是用来定义模板参数关键字,也可以使用class;

typename 后面的T1是随便取的,一般是大写字母或者单词首字母大写;

T1代表是一个模板类型(虚拟类型)

了解了上述格式,我们利用函数模板来将上述交换代码进行一个优化:

#define _CRT_SECURE_NO_WARNINGS 1
#include <iostream>
using namespace std;

//函数模板
template<typename T>
void Swap(T& x, T& y)
{
	T tmp = x;
	x = y;
	y = tmp;
}

int main()
{
	int a1 = 1, b1 = 2;
	Swap(a1, b1);
	float a2 = 1.2, b2 = 3.2;
	Swap(a2, b2);
	char a3 = 'x', b3 = 'y';
	Swap(a3, b3);
	cout << "a1:" << a1 << '\t' << "b1:" << b1 << endl;
	cout << "a2:" << a2 << '\t' << "b2:" << b2 << endl;
	cout << "a3:" << a3 << '\t' << "b3:" << b3 << endl;
	return 0;
}

我们发现确实利用一个模板达到了三组不同类型数据的交换。

那模板到底是怎么起作用的呢?它们调用的都是一个函数吗?那我们看下面:

2.3 函数模板的原理

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

下面给个图更好的对上述语句进行一个直观的解释:

我们发现模板只是一个模板,不是真正的函数。

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

2.4 函数模板的实例化

用不同类型的参数使用函数模板时 ,称为函数模板的 实例化 。模板参数实例化分为: 隐式实
例化显式实例
1.   隐式实例化:让编译器根据实参推演模板参数的实际类型

该语句不能通过编译,因为在编译期间,当编译器看到该实例化时,需要推演其实参类型

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

对于上述问题,我们通常有以下两种解决方案:

1. 用户自己来强制转换;

 2. 重新弄一个模板;

3. 使用显示实例化:在函数名后的<>中指定模板参数的实际类型

2.5 模板参数的匹配原则

1.   一个非模板函数可以和一个同名的函数模板同时存在,而且该函数模板还可以被实例化为
这个非模板函数
// 专门处理int的加法函数
int Add(int left, int right)
{
    returnleft+right;
}

// 通用加法函数
template<class T>
T Add(T left, T right)
{
    return left+right;
}

void Test()
{
    Add(1, 2);       // 与非模板函数匹配,编译器不需要特化
    Add<int>(1, 2);  // 调用编译器特化的Add版本
}
2. 对于非模板函数和同名函数模板,如果其他条件都相同,在调动时会优先调用非模板函数
而不会从该模板产生出一个实例。
如果模板可以产生一个具有更好匹配的函数,那么将选择模板
// 专门处理int的加法函数
int Add(int left, int right)
{
    return left+right;
}

// 通用加法函数
template<class T1, class T2>
T1 Add(T1 left, T2 right)
{
    return left+right;
}

void Test()
{
    Add(1, 2);     // 与非函数模板类型完全匹配,不需要函数模板实例化
    Add(1, 2.0);   // 模板函数可以生成更加匹配的版本,编译器根据实参生成更加匹配的Add函数
}
3. 模板函数不允许自动类型转换,但普通函数可以进行自动类型转换

3. 类模板

3.1 类模板的定义格式

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

为什么会用到类模板定义呢?

在之前学习顺序表、链表...的时候我们通常使用 typedef DataType int 来声明结构里面存的数

据类型,我们想往里面存什么数据就将数据类型重定义不就好了,何必多此一举呢?

其实不然,typedef重定义确实可以达到上述效果,但如果我们想要定义两个链表,每个链表

里面存储不同类型的数据,那是不是就得把上述链表代码写两遍呢?

使用类模板便很好的解决这一问题,我们只要写一份代码,存不同类型数据的时候,编译器

便会自动推演生成存储对应数据类型的类

// 动态顺序表
// 注意: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;
}

我们在使用类模板的时候,不能将类模板的声明和定义放在两个文件中。

如果将类模板声明和定义分开写的话也要写在同一文件中,像上述代码析构函数那种形式。 

3.2 类模板的实例化

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

4. STL简介

4.1 什么是STL

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

4.2 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) 采用,可移植性好, 可公开、修改甚至贩卖,从命名风格和编程 风格上看,阅读性非常高。

4.3 STL的六大组件

4.4  STL的重要性

网上有句话说: 不懂 STL ,不要说你会 C++” STL C++ 中的优秀作品,有了它的陪伴,许
多底层的数据结构 以及算法都不需要自己重新造轮子,站在前人的肩膀上,健步如飞的快速
开发。

4.5 如何学习STL

 简单总结一下:学习STL的三个境界:能用,明理,能扩展

4.6 STL的缺陷

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

坚持打卡!😃

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

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

相关文章

基于.Net开发的ChatGPT客户端,兼容Windows、IOS、安卓、MacOS、Linux

2023年目前要说最热的点,肯定是ChatGPT了。 ChatGPT官方提供的网页版本,还有需要科学上网,很多人都会基于此进行封装。 现在是移动互联网时代,基于手机APP的需求还是很大的。 所以,今天给大家推荐一个ChatGPT客户端开源项目,兼容苹果和安卓手机、PC。 项目简介 这是…

若依管理系统修改页面标题和logo

一&#xff1a;修改网页上的logo 把public目录下favicon.ico&#xff0c;换成自己logo 注&#xff1a;替换图片的名字最好还是以favicon.ico命名&#xff0c;如果改变,就要改public目录下的index.html代码 二&#xff1a;修改页面上的logo 把src/assets/logo/logo.png换成自己l…

【Chano的SFM教程】3dmax 面部表情.VTA基本制作教程

本篇教程作者为&#xff1a;小鸟Chano&#xff0c;转载请表明作者和出处&#xff1a;CSDN 欢迎观看本次教程 本教程将会为你演示使用3D MAX 制作一个基本的SFM表情控制器【表情滑条】并导入SFM进行使用。 Chano自己也是近期才掌握的这项知识&#xff0c;所以过程中可能有很多…

GDB调试实验

一、实验准备 在 Linux 环境软件开发中&#xff0c;GDB 是调试 C 和 C 程序的主要工具。本次实验围绕着GDB常用的调试操作进行。 1、设置断点的意义 当我们想查看变量内容&#xff0c;堆栈情况等等&#xff0c;可以指定断点。程序执行到断点处会暂停执行。break 命令用来设置…

React 条件渲染组件

组件通常需要根据不同的条件显示不同的内容&#xff0c;以及根据应用的状态变化只渲染其中的一部分。 在 React 中&#xff0c;可以使用 JavaScript 语法有条件地呈现 JSX&#xff0c;比如 if 语句、&&和 ?: 操作符。 根据条件返回 JSX Demo.js 文件&#xff1a; …

【高危】vm2 <3.9.17 沙箱逃逸漏洞(POC)(CVE-2023-30547 )

漏洞描述 vm2 是一个基于 Node.js 的沙箱环境&#xff0c;可以使用列入白名单的 Node 内置模块运行不受信任的代码。 由于 CVE-2023-29199 的修复不完整&#xff0c;vm2 3.9.17 之前版本的 transformer.js 文件中的 transformer 函数异常处理逻辑存在缺陷。攻击者可以利用这个…

【01-Java Web先导课】-如何进行JDK的安装(或Java环境的配置)

文章目录 一、JDK&#xff08;jdk-8u371-windows-x64.exe&#xff09;的下载1、下载网址2、选择相应版本进行下载 二、JDK&#xff08;jdk-8u371-windows-x64.exe&#xff09;的安装1、JDK安装2、系统环境变量配置3、classspath环境变量设置 免责说明 一、JDK&#xff08;jdk-8…

❤echarts 南丁格尔玫瑰图的使用以及南丁格尔玫瑰图详细配置

❤echarts 南丁格尔玫瑰图的使用以及南丁格尔玫瑰图详细配置 1、认识 使用可以参考之前文章&#xff0c;会使用直接跳过1 引入官网的南丁格尔玫瑰图效果如下&#xff1a; 使用函数配置分为三个部分&#xff1a;初始化> 设置配置> 地图使用参数 配置代码如下 option…

法学考生必看—外经贸法学专业在职研究生

法学专业就业面比较广&#xff0c;但很多人工作后都觉得还是缺少核心竞争力&#xff0c;想通过读研来改变现状&#xff0c;23考研已经落幕&#xff0c;想要今年就能入班学习的院校有吗&#xff1f;有法学专业的吗&#xff1f; 一、学校介绍 对外经济贸易大学创建于1951年&…

STM32 USB资料整理

CypressUSB中文文档 https://img.anfulai.cn/bbs/90026/AN57294%20USB%20101%20An%20Introduction%20to%20Universal%20Serial%20Bus%202.0%20(Chinese).pdf RL-USB教程 https://www.armbbs.cn/forum.php?modviewthread&tid99710 USB应用实战教程第3期&#xff1a;手把…

Java Jvm中的垃圾回收机制

jvm的垃圾回收机制是什么 jvm的垃圾回收机制是GC&#xff08;Garbage Collection&#xff09;&#xff0c;也叫垃圾收集器。 GC基本原理&#xff1a;将内存中不再被使用的对象进行回收&#xff1b;GC中用于回收的方法称为收集器&#xff0c;由于GC需要消耗一些资源和时间&…

认识BASH这个Shell

文章目录 认识BASH这个Shell硬件、内核与shell为什么要学命令行模式的Shell&#xff1f;Bash Shell的功能命令与文件补全(TAB)命令别名设置(alias)历史命令(history)任务管理、前台、后台控制(jobs&#xff0c;fg&#xff0c;bg)通配符程序化脚本 查询命令是否为Bash shell 的内…

前端--移动端布局--2移动开发之flex布局

目标&#xff1a; 能够说出flex盒子的布局原理 能够使用flex布局的常用属性 能够独立完成携程移动端首页案例 目录&#xff1a; flex布局体验 flex布局原理 flex布局父项常见属性 flex布局子项常见属性 写出网首页案例制作 1.flex布局体验 1.1传统布局与flex布局 传…

nacos 部署 配置

文章目录 一、Nacos简介 1、为什么叫Nacos2、Nacos是什么3、能干嘛4、去哪下二、安装并运行Nacos 2.1 基础环境及配置&#xff1a;三、Nacos作为服务注册中心演示 3.1 基于Nacos的服务提供者 2、POM文件3、YML文件4、主启动5、业务类6、测试7、nacos控制台3.2 基于Nacos的服务消…

[POJ - 1080 ]Palindrome(区间DP)

[POJ - 1080 ]Palindrome&#xff08; 区间DP&#xff09; 1、问题2、分析状态表示状态转移空间优化 3、代码 1、问题 给定一个字符串&#xff0c;通过添加最少的字符&#xff0c;使得这个字符串成为一个回文字符串。 2、分析 状态表示 f [ i ] [ j ] f[i][j] f[i][j]表示将…

2023年产品经理需要考的证书,NPDP含金量真高

产品经理国际资格认证NPDP是新产品开发方面的认证&#xff0c;集理论、方法与实践为一体的全方位的知识体系&#xff0c;为公司组织层级进行规划、决策、执行提供良好的方法体系支撑。 【认证机构】 产品开发与管理协会&#xff08;PDMA&#xff09;成立于1979年&#xff0c;是…

盖子的c++小课堂——第十七讲:递归

前言 通知一下&#xff0c;以后每周不定期更新&#xff0c;有可能是周六更新&#xff0c;也可能是周日吧&#xff0c;反正会更新的~~还有我新出的专栏《跟着盖子读论语》&#xff0c;记得订阅一下啊跟着盖子学《论语》_我叫盖子的盖鸭的博客-CSDN博客 三元表达式 三元表达式…

单链表中二级指针的使用以及单链表的增删查改操作

前言&#xff1a;在链表的构建中,链表的初始化和销毁为何需要使用一个二级指针,而不是只需要传递一个指针就可以了,其问题的关键就在于c语言的参数传递的方式是值传递 那么,这篇文章就来聊一聊在链表的初始化中一级指针的传递和二级指针的区别&#xff0c;并总结给出单链表的C语…

AutoCAD使用技巧

AutoCAD使用技巧 环境说明AutoCAD 导入EXCELAutoCAD移动对象到原点 环境说明 本文基于AutoCAD 2021版本。 AutoCAD 导入EXCEL 如果菜单栏隐藏点击显示菜单栏&#xff1a; 在excel中选中copy内容赋值&#xff0c;AutoCAD中选择&#xff1a;编辑-选择性粘贴-作为AutoCAD图元…

必用WhatsApp营销的4个理由

WhatsApp是世界上最受欢迎的消息传递应用程序。每天有1万新用户加入WhatsApp。各种规模的公司都利用该平台与世界各地的客户进行有效的沟通&#xff0c;这要归功于其广泛的覆盖范围、用户友好的设计和安全的端到端加密。因此&#xff0c;WhatsApp聊天机器人迅速普及。 1.为您的…