【C++进阶(二)】STL大法--vector的深度剖析以及模拟实现

💓博主CSDN主页:杭电码农-NEO💓

⏩专栏分类:C++从入门到精通⏪

🚚代码仓库:NEO的学习日记🚚

🌹关注我🫵带你学习C++
  🔝🔝


在这里插入图片描述

vector

  • 1. 前言
  • 2. 熟悉vector的接口函数
    • 2.1 vector的构造与拷贝构造
    • 2.2 vector迭代器的使用
    • 2.3 vector空间相关函数
    • 2.4 vector的增删查改
      • 2.41 find,swap和sort
      • 2.42 insert和erase
      • 2.43 随机访问operator[ ]
  • 3. vector的模拟实现
    • 3.1 vector容量相关函数
    • 3.11 reverse函数
      • 3.12 resize函数
    • 3.2 vector的构造函数
    • 3.3 vector的析构函数
    • 3.4 vector的拷贝构造函数
  • 4. 总结以及拓展

1. 前言

和string的学习不同
vector即要掌握它的用法
更要会自己去实现一个vector

本章重点:

熟悉STL库中vector的接口函数
自己实现一个简易vector类
本章只实现容量相关函数
和构造,析构,拷贝构造函数

注:vector其实就是顺序容器
string类只用考虑存储字符
然而vector中可以存储任一类型
所以vector的自我实现需要用模板


2. 熟悉vector的接口函数

还是借助老朋友:cplusplus来查阅文档

在这里插入图片描述

库中的vector的模板参数有两个
后一个是内存池,用来提升空间利用效率
对于现阶段的学习而言可有可无


2.1 vector的构造与拷贝构造

在这里插入图片描述

常见的构造有:

vector<int> v1;
vector<int> v2(10,1);
vector<int> v3(v2);

v2:构造并初始化10个值为1的顺序表

vector可以用迭代器区间初始化:

string str("abcdefg");
vector<string> vv(str.begin(),str.end());

2.2 vector迭代器的使用

在这里插入图片描述
和string一样,vector有正向和反向
两种迭代器,且使用方法和string相同

在这里插入图片描述

vector<int> vv{1,2,3,4,5,6};
vector<int>::iterator it = vv.begin();
while(it!=vv.end())
{cout<<*it;it++;
}

2.3 vector空间相关函数

在这里插入图片描述

vector的空间相关的函数
和string的机会一模一样
如果你看了文档还不懂的话
可以先阅读此篇文章:string接口函数


2.4 vector的增删查改

在这里插入图片描述
push_back和pop_back
都是老朋友了,这里就不多说了
在介绍insert和erase之前
先来了解几个算法库的函数


2.41 find,swap和sort

这三个函数都在头文件:algorithm

在这里插入图片描述

find函数:参数是一段迭代器区间
以及在此区间你需要查找的值
找到后返回这个值对应的迭代器位置
若找不到,则返回迭代器last

find的使用:

vector<int> vv{1,2,3,4,5,6,7,8,9};
auto pos = find(vv.begin(),vv.end(),5);
cout<<*pos;

注:使用auto是为了简写迭代器也可以用
vector< int >::iterator替代

在这里插入图片描述

swap想必是大家的常客了
这里给它个面子,就不介绍它了

在这里插入图片描述

sort非常方便,它内部实现是快排
我们只需要传一个迭代器区间
就可以将整个区间排好序

sort的使用:

vector<int> vv{5,7,3,9,6,4,1,2,8};
sort(vv.begin().vv.end());

2.42 insert和erase

在这里插入图片描述

和string不同,vector的insert
的参数pos不是整型,而是迭代器
默认是在pos位置前插入一个数据

insert和find常常配合在一起使用

在整型5前面插入一个100:

vector<int> vv{1,2,3,4,5,6,7,8,9};
auto pos = find(vv.begin(),vv.end(),5);
vv.insert(pos,100);

在这里插入图片描述

和string的erase不同,vector
的erase一次只删除一个数据
然而string如果使用缺省值就是
将全部数据删完

vector的erase甚至可以删除一段区间

删除顺序表中值为100的元素

vector<int> vv{1,2,3,4,5,6,7,8,9,100};
auto pos = find(vv.begin(),vv.end(),100);
vv.erase(pos);
//删除一个区间
vv.erase(vv.begin()+2,vv.end()-2);

2.43 随机访问operator[ ]

vector中最喜欢用的是[ ]
它支持随机访问,是否方便

operator[]的使用:

vector<int> vv{1,2,3,4,5,6,7,8,9};
for(int i=0;i<vv.size();i++)
{cout<<vv[i]<<" ";
}

3. vector的模拟实现

首先要关注的是成员变量
vector是顺序表,所以和实现C语言
时的顺序表一样,至少有三个参数

  1. 指向一段空间的指针
  2. 空间的有效大小
  3. 空间的实际大小

由于vector的迭代器就是普通指针
所以成员变量的类型其实是迭代器

template<class T>
class vector
{
public:typedef T* iterator;private:iterator _start;iterator _finish;iterator _endof_storage;

这里使用迭代器作为三个参数的类型
是因为:求vector的size和capacity时
可以直接使用finish-start
也就是指针相减求出长度

成员变量和空间的关系:

在这里插入图片描述


3.1 vector容量相关函数

上来首先要考虑的容量相关的函数:

  • size
  • capacity
  • empty
  • resize
  • reverse

前三个十分简单:

size_t size() const
{return _finish - _start;
}size_t capacity() const
{return _endof_storage - _finish;
}bool empty() const
{return (size()==0);
}

3.11 reverse函数

reverse只会改变capacity的大小
并不会改变size的大小

void reserve(size_t n)
{if (n > capacity()){size_t sz = size();T* tmp = new T[n];if (_start){//memcpy(tmp, _start, sizeof(T)*sz);for (size_t i = 0; i < sz; ++i){tmp[i] = _start[i];}delete[] _start;}_start = tmp;_finish = _start + sz;_end_of_storage = _start + n;}
}

注:当n小于capacity时,不进行扩容

由于C++内存管理的new
无法像C语言的realloc一样原地扩容
所以必须先开辟n个空间,再将数据
拷贝到新空间,且释放旧空间


3.12 resize函数

resize即会改变size大小
也会改变capacity大小

resize要分三种情况:

  1. n大于capacity时
  2. n大于size小于capacity时
  3. n小于size时

它们的解决方案分别是:

  • 直接套用reversezhu

  • 初始有效值不变,在此之后
    初始化新的内容

  • 直接将size缩小到n

void resize(size_t n, const T& val = T())
{if (n > capacity()){reserve(n);}if (n > size()){// 初始化填值while (_finish < _start + n){*_finish = val;++_finish;}}else{_finish = _start + n;}
}

注:参数val=T()使用了匿名对象
C++将内置类型特殊处理过
int/char等等都被升级为了类
所以可以使用int()表示匿名对象

int tmp1 = int();
int tmp2 = int(10);

int的缺省值为0


3.2 vector的构造函数

  1. 首先最简单的无参构造:
vector():_start(nullptr), _finish(nullptr), _end_of_storage(nullptr)
{}
  1. 紧接着是带参的构造函数
    我们跟着STL库的风格走:
vector(size_t n, const T& val = T()):_start(nullptr), _finish(nullptr), _end_of_storage(nullptr)
{reserve(n);//开辟n个空间for (size_t i = 0; i < n; ++i){push_back(val);//给初始值赋值}
}
  1. 最后是使用迭代器区间来构造
    比如我想在顺序表中存放string类型:
string str("abcdefg");
vector<string> vv(str.begin(),str.end());

此时在模板类中还应该有一个模板

template <class InputIterator>
vector(InputIterator first, InputIterator last):_start(nullptr), _finish(nullptr), _end_of_storage(nullptr)
{while (first != last){push_back(*first);++first;}
}

注:inputiterator取名是模仿STL的
你也可以取任一除了T的名字


3.3 vector的析构函数

vector的析构函数非常简单
只需要将空间释放
并且将各个指针置为空就行了

~vector()
{delete[] _start;_start = _finish = _end_of_storage = nullptr;
}

3.4 vector的拷贝构造函数

拷贝构造的实现有很多种写法
大家可以先自己尝试一下

vector(const vector<T>& v):_start(nullptr), _finish(nullptr), _endof_storage(nullptr){reserve(v.size());for (const auto& e : v){push_back(e);}}

4. 总结以及拓展

vector模拟实现的全部代码我将在
下一篇文章中分享给大家

可以发现:STL的神奇之处在于
它把所有接口函数都做了统一化处理
每一个容器的接口函数的使用都相似
但是内部实现被这种封装隐藏起来了
进一步又体现了C++的三大特性:
封装

并且C++实现了所有容器通用的算法库
比如sort和find都只需要传迭代器
然而所有容器都会被迭代器封装
所以一份代码就能实现对不同容器的操作

在这里插入图片描述

拓展题目:

熟悉了vector的基本使用
可以尝试解决一下下面几个问题:

  • 只出现一次的数字
  • 删除有序数组中的重复项
  • 数组中出现次数超过一半的数字
  • 杨辉三角

留给大家当作小试牛刀了~


🔎 下期预告:迭代器失效和深浅拷贝问题 🔍

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

R语言响应面(RSM)、线性模型lm分析生产过程影响因素可视化

全文链接&#xff1a;https://tecdat.cn/?p33499 响应面&#xff08;Response Surface Methodology&#xff0c;RSM&#xff09;分析是一种常用的统计方法&#xff0c;用于研究和优化生产过程中的影响因素。通过建立数学模型来描述因素与响应之间的关系&#xff0c;RSM可以帮助…

用Cmake build OpenCV后,在VS中查看OpenCV源码的方法(环境VS2022+openCV4.8.0) Part III

用Cmake build OpenCV后&#xff0c;在VS中查看OpenCV源码的方法(环境VS2022openCV4.8.0) Part III 用Cmake build OpenCV后&#xff0c;在VS中查看OpenCV源码的方法&#xff08;环境VS2022openCV4.8.0&#xff09; Part I_松下J27的博客-CSDN博客 用Cmake build OpenCV后&…

③matlab向量和矩阵

目录 手动输入数组 创建等间距向量 数组创建函数 手动输入数组 1.背景 单个称为标量的数值实际上是一个 11 数组&#xff0c;也即它包含 1 行 1 列。 任务 创建一个名为 x 并且值为 4 的变量。 2.您可以使用方括号创建包含多个元素的数组。 x [3 5] x 3 5 任务 …

Python钢筋混凝土结构计算.pdf-混凝土构件计算

计算原理&#xff1a; 代码实现&#xff1a; #钢筋混凝土参数 def c_hrb(): global fcuk,HRB,Ec,fc,ft,ftk,Es,fy,fyp,fyk global a1,epsilon_cu fcukEcfcftftk0.0 HRBEsfyfypfyk0.0 #矩形应力图系数a1&#xff0c;C50以下为1.0 a11.0 #正截面混凝土极限压应变epsilon_cu&#…

uni-app+uView实现点击查看大图片的效果

<u-button text"月落" click"imgPreview()"></u-button> //注意&#xff1a;参数urls 是预览图片的链接地址&#xff0c;是个数组 imgPreview() {uni.previewImage({indicator: "none",loop: false,urls: []&#xff0c;}) },参数说…

纵行科技与山鹰绿能达成合作,提供物联网资产管理数据服务

近日&#xff0c;纵行科技与山鹰绿能宣布双方达成深度合作关系&#xff0c;纵行科技将为山鹰绿能提供专业的物联网技术服务&#xff0c;使用物联网技术帮助山鹰绿能对循环包装载具等资产进行在线管理和数字化运营。 据悉&#xff0c;山鹰绿能是一家由山鹰国际控股的全资子公司…

Kafka3.0.0版本——Follower故障处理细节原理

目录 一、服务器信息二、服务器基本信息及相关概念2.1、服务器基本信息2.2、LEO的概念2.3、HW的概念 三、Follower故障处理细节 一、服务器信息 三台服务器 原始服务器名称原始服务器ip节点centos7虚拟机1192.168.136.27broker0centos7虚拟机2192.168.136.28broker1centos7虚拟…

中国智慧燃气行业市场需求

文章来源&#xff1a;中研普华产业研究院 关键词&#xff1a;智慧燃气、智慧燃气场站、智慧燃气平台、设备设施数字化、数字孪生、工业互联网 智慧燃气&#xff0c;是以城市输气管网为基础&#xff0c;各终端用户协调发展&#xff0c;以信息通信平台为支撑&#xff0c;具有信…

JWT 技术的使用

应用场景&#xff1a;访问某些页面&#xff0c;需要用户进行登录&#xff0c;那我们如何知道用户有没有登录呢&#xff0c;这时我们就可以使用jwt技术。用户输入的账号和密码正确的情况下&#xff0c;后端根据用户的唯一id生成一个独一无二的token&#xff0c;并返回给前端&…

Ansible自动化运维工具(二)

目录 &#xff08;6&#xff09;copy模块 &#xff08;7&#xff09;file模块 ​编辑​编辑&#xff08;8&#xff09;hostname模块 &#xff08;9&#xff09;ping模块 &#xff08;10&#xff09;yum 模块 &#xff08;11&#xff09;service/system模块 ​编辑 ​…

解决Android Studio中Plugin version和Gradle version不匹配的问题

生命中最艰难的那段路是要自己一个人走过来的&#xff0c;这样&#xff0c;学到更多的是坚强&#xff0c;而不是感动。 《红猪》 前言 导入一个百度云的Demo而已&#xff0c;居然遇到这么多问题&#xff0c;纠结了很久&#xff0c;也查了很多资料&#xff0c;弯弯绕绕了好多路…

node.js 简单实验 创建一个简单的web服务

概要&#xff1a;用一个最简单是例子感受一下node.js 的能力 1.代码 var http require("http") http.createServer(function (request, response) { response.writeHead(200, {Content-Type: text/plain}); response.end(Hello World\n); }).listen(8081); cons…

TCP/UDP原理

文章目录 一、端口1. 端口的定义和作用2.服务端和客户端的区别3.常见的知名端口号有 二、TCP的原理1.TCP头部封装格式2.TCP可靠性机制三次握手确认机制四次挥手RST结束连接窗口机制 3.完整性校验4.TCP特征5.TCP的适用场景 三、UDP的原理1.UDP头部封装格式2.UDP特征3.UDP的适用场…

Java 数据结构使用学习

Set和List的区别 Set 接口实例存储的是无序的&#xff0c;不重复的数据。List 接口实例存储的是有序的&#xff0c;可以重复的元素。 Set 检索效率低下&#xff0c;删除和插入效率高&#xff0c;插入和删除不会引起元素位置改变 <实现类有HashSet,TreeSet>。 List 和数…

Fegin异步情况丢失上下文问题

在微服务的开发中&#xff0c;我们经常需要服务之间的调用&#xff0c;并且为了提高效率使用异步的方式进行服务之间的调用&#xff0c;在这种异步的调用情况下会有一个严重的问题&#xff0c;丢失上文下 通过以上图片可以看出异步丢失上下文的原因是不在同一个线程&#xff0c…

敲打美国?纯国产5G芯片推出,美国的打压取得了反效果

就在美国商务部长雷蒙多访华前&#xff0c;一家国产手机企业突然宣布国产5G手机推售&#xff0c;在这特别的时间点&#xff0c;或许是为了敲打美国&#xff0c;说明中国芯片已取得巨大的进步&#xff0c;国产芯片产业链已得到进一步的完善。 这家国产手机企业这几年一直都备受美…

加速通导融合,中国在精准定位领域脱颖而出

近日&#xff0c;上海正式发布“5G揽海”行动计划&#xff0c;旨在构建陆海空天一体化海洋网络&#xff0c;加快建设基于“北斗5G”的超高精定位网的海洋新型基础设施&#xff0c;赋能数字经济时代下航运的高质量发展。 这是中国数字经济蓬勃发展下的一个小缩影。今年以来&…

运维高级学习--Kubernetes(K8s 1.28.x)部署

一、基础环境配置&#xff08;所有主机操作&#xff09; 主机名规划 序号 主机ip 主机名规划1 192.168.1.30 kubernetes-master.openlab.cn kubernetes-master2 192.168.1.31 kubernetes-node1.openlab.cn kubernetes-node13 192.168.1.32 kubernetes-node2…

【Terraform学习】使用 Terraform创建DynamoDB添加项目(Terraform-AWS最佳实战学习)

本站以分享各种运维经验和运维所需要的技能为主 《python》&#xff1a;python零基础入门学习 《shell》&#xff1a;shell学习 《terraform》持续更新中&#xff1a;terraform_Aws学习零基础入门到最佳实战 《k8》暂未更新 《docker学习》暂未更新 《ceph学习》ceph日常问题解…

中国应试教育市场:挑战与机遇并存,金榜状元引领前行

2023年全国高考报名人数1291万人再次刷新历史纪录&#xff0c;但一本的录取率仅为23%&#xff1b;教育部2021年开始推行中考分流政策&#xff0c;只有约为50%初中毕业生可以升入普通高中&#xff1b;“双减”政策的推行&#xff0c;使得高考升学的压力提前到中考阶段&#xff0…