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

【C++】模版初阶以及STL的简介

在这里插入图片描述
个人主页~


模版及STL

  • 一、模版初阶
    • 1、泛型编程
    • 2、函数模版
      • (1)概念
      • (2)函数模版格式
      • (3)函数模版的原理
      • (4)函数模版的实例化
        • ①显式实例化
        • ②隐式实例化
      • (5)模版参数的匹配原则
    • 3、类模版
      • (1)类模板的定义格式
      • (2)类模板的实例化以及类函数的使用
  • 二、STL简介

一、模版初阶

1、泛型编程

我们在先前的博文中提到过函数重载交换函数swap,但是使用函数重载有几个不好的地方:
1、重载的函数仅仅是类型不同,代码复用率比较低,所有的重载函数中只有数据类型不同,其他的都基本相同, 只要有新类型出现就需要用户自己增加对应的函数
2、代码可维护性低,其中某一函数出错可能会导致所有重载都出错

解决这个问题的方法就是有一个模具,只要相同的就直接套用,不同的替换就可以了

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

2、函数模版

(1)概念

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

(2)函数模版格式

template<typename T1,typename T2,...>
type name ()
{}

template就是一个函数模版关键字后跟<>,参数放里边
typename是一个类类型参数,也可以写成class,但不能写成struct,它代表name函数可以操作的数据类型。当调用name函数时,编译器会根据传递给函数的实参类型来推断T的具体类型
type name(){}就是一个函数

具体看这个例子:

template<typename T>
void Swap( T& left, T& right)
{T temp = left;left = right;right = temp;
}

(3)函数模版的原理

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

在编译器的编译阶段,编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用,将T确定为某一具体类型,然后产生一份专门处理该类型的代码

(4)函数模版的实例化

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

①显式实例化
template<typename T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{int a = 10;double b = 20.0;// 显式实例化Add<int>(a, b);return 0;
}

格式:函数名+<数据类型>(参数)

②隐式实例化
template<typename T>
T Add(const T& left, const T& right)
{return left + right;
}
int main()
{int a1 = 10, a2 = 20;double d1 = 10.0, d2 = 20.0;Add(a1, a2);//意味着T实例化为intAdd(d1, d2);//意味着T实例化为doublereturn 0;
}

在这里插入图片描述
这是当两个参数类型相同的时候,如果两参数类型不同该语句不能通过编译,因为在编译期间,当编译器看到该实例化时,需要推演其实参类型,通过实参a1将T推演为int,通过实参d1将T推演为double类型,但模板参数列表中只有一个T,编译器无法确定此处到底该将T确定为int 或者 double类型而报错,所以我们要进行手动强制类型转换
在这里插入图片描述
在这里插入图片描述

注意:在模板中,编译器一般不会进行类型转换操作

(5)模版参数的匹配原则

①一个非模版函数可以和一个同名的函数模版同时存在,而且该函数模版还可以被实例化为这个非模版函数

int Add(int left, int right)
{return left + right;
}template<typename T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{cout << Add(1, 2) << endl;//直接与非模版函数相匹配,编译器不需要进行特化cout << Add<int>(1, 2) << endl;//调用编译器特化的Add版本return 0;
}

在这里插入图片描述

调试结果:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
②对于非模版函数和同名函数模版,如果其他条件都相同,在调动时会优先调用非模版函数而不会从该模版产生出一个实例,但如果模版可以产生一个具有更好的匹配的函数,那么将选择模版

int main()
{Add(1, 2);//非模版更匹配,会直接选择非模版函数,不必再特化Add(1, 2.0);//模版生成的函数比非模版函数更加匹配,这样就会选择模版return 0;
}

③普通函数可以进行自动类型转换,模版函数不允许自动类型转换

3、类模版

(1)类模板的定义格式

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

name是类模板名

接下来我们写一个动态顺序表的模版

template<class T>
class Vector//Vector是类模版名,不是类
{
public:Vector(size_t capacity = 10): _pData(new T[capacity]), _size(0), _capacity(capacity){}
//模版外定义函数~Vector();void PushBack(const T& data);
//模版内定义函数void PopBack(){_size--;}size_t Size(){return _size;}private:T* _pData;size_t _size;size_t _capacity;
};
// 类模板中函数放在类外进行定义时,需要加模板参数列表
template <class T>
Vector<T>::~Vector()
{if (_pData)delete[] _pData;_size = _capacity = 0;
}
//每次函数定义都要加模版参数列表
template <class T>
void Vector<T>::PushBack(const T& data)
{if (_size == _capacity){size_t newcapacity = _capacity * 2;T* ptr = new T[newcapacity];for (int i = 0; i < _size; i++){ptr[i] = _pData[i];}delete[] _pData;//用delete[]回收new[]申请的空间_pData = ptr;_capacity = newcapacity;}_pData[_size] = data;_size++;
}

(2)类模板的实例化以及类函数的使用

类模板实例化需要在类模板名字后加<>,然后将实例化的类型放在里边,类模板不是类,实例化后才为类

int main()
{Vector<int> s1;s1.PushBack(1);s1.PushBack(2);s1.PushBack(3);s1.PushBack(4);s1.PopBack();cout << s1.Size() << endl;Vector<double> s2;return 0;
}

在这里插入图片描述
调试结果:
在这里插入图片描述
在这里插入图片描述

二、STL简介

STL是标准库的组成部分,是一个可复用的组件库和包罗数据结构与算法的软件框架

STL有四个版本,分别是惠普版本(也叫原始版本)、P.J.版本(VC)、RW版本、SGI版本(Linux)

STL由容器、算法、仿函数、空间配置器、迭代器、配接器六大组件构成

STL的产生是C++的一次巨变,它使得很多底层的数据结构及算法不用再让程序员来实现,大大提高了学习和工作的效率和开发产品的进度

当然,STL也有更新慢、不支持线程安全、内部复杂、代码膨胀等问题


今日分享到此结束~
在这里插入图片描述

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

相关文章:

  • 51单片机学习(4)
  • 3D问界—MAYA制作铁丝栅栏(透明贴图法)
  • 编译器对C++23的支持程度
  • k8s核心操作_存储抽象_K8S中使用Secret功能来存储密码_使用免密拉取镜像_k8s核心实战总结---分布式云原生部署架构搭建033
  • 21集 ESP32-IDF开发教程-《MCU嵌入式AI开发笔记》
  • 《大数据基础》相关知识点及考点,例题
  • 网络通信介绍
  • 16、Python之容器:元组与列表、推导式与生成式,差之毫厘谬以千里
  • HTTP协议——请求头和请求体详情
  • 编程中的智慧之设计模式二
  • 基于python的百度资讯爬虫的设计与实现
  • 用 WireShark 抓住 TCP
  • Lua基础知识入门
  • 【机器学习实战】Datawhale夏令营2:深度学习回顾
  • 开发扫地机器人系统时无法兼容手机解决方案
  • Elasticsearch 角色和权限管理
  • 华为HCIP Datacom H12-821 卷42
  • 【精品资料】物业行业BI大数据解决方案(43页PPT)
  • 推荐一款处理TCP数据的架构--EasyTcp4Net
  • 2、电脑各部件品牌介绍 - 计算机硬件品牌系列文章
  • Git【撤销远程提交记录】
  • java基础学习:序列化之 - Fast serialization
  • Microsoft Build 2024 推出 .NET 9:Tensor<T>、 OpenAI Collaboration和.NET Aspire
  • 【Neural signal processing and analysis zero to hero】- 2
  • 好用的AI搜索引擎
  • 十、Java集合 ★ ✔(模块18-20)【泛型、通配符、List、Set、TreeSet、自然排序和比较器排序、Collections、可变参数、Map】
  • 阿里云开源 Qwen2-Audio 音频聊天和预训练大型音频语言模型
  • SpringBoot集成MQTT实现交互服务通信
  • python实现插入排序、快速排序
  • Spring Boot集成kudu快速入门Demo