C++STL详解(三)——vector的介绍和使用

news2025/1/22 16:53:44

文章目录

  • vector的介绍
  • vector的使用
    • vector的定义方式
    • vector的空间增长问题
      • reserve和resize
    • vector的迭代器使用
      • begin 和end
      • rbegin和rend
      • insert 和erase
      • find函数
      • 元素访问
    • vector迭代器失效问题
      • 1:inserse插入扩容时空间销毁造成野指针问题
      • 2:erase删除或者insert插入时元素移动迭代器问题:

vector的介绍

1: vector是表示可变大小数组的容器。
2:vector就像数组一样,也采用的连续空间来存储元素,这也意味着vector可以采用下标对vector的元素进行访问。
3:vector与普通数组不同的是,vector的大小是可以动态改变的。
4:vector需要分配大小时,其做法是,分配一个新的数组,然后将全部元素移入到这个数组当中,并且释放原来的空间。
5:由于vector采用连续的空间来存储元素,与其他动态序列容器相比,vector在访问元素的时候更加高效,在其末尾添加和删除元素相对高效,而对于不在其末尾进行的删除和插入操作效率则相对较低。

vector的使用

vector的定义方式

方式一:构造一个vector类型的容器。

vector<int> v1;   

方式二: 构造一个含有n各val的vector容器

vector<int> v2(10, 2);

方式三:vector容器的拷贝构造

vector<int>v3(v2);

方式四:使用迭代器区间构造(该方式也可用于构造其他容器,例如string类型。)

vector<int> v4(v2.begin(), v2.end());

方式6:像C语言定义数组一样定义。

vector<int> v5{ 1,2,3,4,5};

vector的空间增长问题

reserve和resize

reserve说明
1:改变容器的最大的最大容量,当我们所传的值大于容器当前的
capacity时,会将capacity扩大到该值。
2:当所给值小于容器当前的capacity是,reserve无效果。
resize说明
1:当所传值大于容器当前的size时,会将size扩大到所传值,扩大的元素为第二个所传值,如果用户没有给出,则编译器会给上缺省值为0;
2:当所传值小于容器当前的size时,则会将vector容器的size缩小到所传值大小。

int main()
{
	vector<int> v(10, 2);
	v.resize(15);//扩容,并让size()大小为15,5个0用来填充。
	for (size_t i = 0; i < v.size(); ++i)
	{
		cout << v[i] << " ";
	}
	cout << endl;
	v.resize(10); //让size()大小变为10,vector容量不变。
	for (size_t i = 0; i < v.size(); ++i)
	{
		cout << v[i] << " ";
	}
	cout << v.capacity() << endl; //20
	return 0;
}

vector的迭代器使用

在这里插入图片描述

begin 和end

int main()
{
	vector <int>v(10, 6);
	vector<int>::iterator it = v.begin();
	while (it != v.end())
	{
		cout << *it << endl;
		it++;
	}
	cout << endl;
}

rbegin和rend

int main()
{
	vector<int> v{ 1,2,3,4,5,6 };
	//反向迭代器遍历容器:reverse_iterator
	vector<int>::reverse_iterator rit = v.rbegin();
	while (rit != v.rend())
	{
		cout << *rit << " ";
		rit++;
	}
	cout << endl;
	return 0;
}

insert 和erase

insert函数可以在目标位置插入1个或者多个指定元素。
erase函数可以删除迭代器指定位置的元素,也可以使用迭代器
进行删除。(左闭右开)

int main()
{
	vector<int> v;
	v.push_back(1); 
	v.push_back(2); 
	v.push_back(3); 
	v.push_back(4); 

	v.pop_back();

	//插入位置,插入个数,插入值。
	v.insert(v.begin(), 3,1);
	for (size_t i = 0; i < v.size(); ++i)
	{
		cout << v[i] << " ";
	}
	cout << endl;
	
	v.erase(v.begin(), v.begin() + 1);
	for (size_t i = 0; i < v.size(); ++i)
	{
		cout << v[i] << " ";
	}
	cout << endl;
	
	return 0;
}

find函数

find函数说明:
使用find函数要确认所要删除位置的迭代器区间,第三个参数则要确定用户所要寻找的值。
如果find函数在所传的迭代器区间找到了目标元素,则返回目标元素迭代器,否则则返回end()位置迭代器。


int main()
{
	vector<int> v;
	v.push_back(1); //尾插元素1
	v.push_back(2); //尾插元素2
	v.push_back(3); //尾插元素3
	v.push_back(4);
	v.push_back(5);


	vector<int>::iterator it = v.begin();

	//在区间寻找值为2的元素,并返回对应迭代器。
	//auto 根据后面的返回值类型主动判断。
	auto  pos = find(it, it + 4, 2);
	
	if ( pos != v,end())
	{
	//删除pos所指的元素。
	v.erase(pos);
	}
	for (size_t i = 0; i < v.size(); ++i)
	{
		cout << v[i] << " ";
	}
	cout << endl;
	return 0;
}

元素访问

vector中除了使用vector迭代器进行访问的,还可以使用[]操作符重载访问。

int main()
{
	vector<int> v(10, 1);
	//使用“下标+[]”的方式遍历容器
	for (size_t i = 0; i < v.size(); i++)
	{
		cout << v[i] << " ";
	}
	cout << endl;
	return 0;
}

另外,vector还支持迭代器,这也就说明vector还支持范围for进行访问。

int main()
{
	vector<int> v(10, 6);
	for (auto e : v)
	{
		cout << e << " ";
	}
	cout << endl;
	return 0;
}

vector迭代器失效问题

1:inserse插入扩容时空间销毁造成野指针问题

例如,当我们从vector容器3的位置前插入30,不对vector进行扩容时,发现程序正常运行。
但是当我们插入数据时要对vector进行扩容却发现程序而不能崩溃了。

void test_vector1()
{
	vector <int> v;
	v.push_back(1);
	v.push_back(2);
	v.push_back(3);
	v.push_back(4);
	for (auto e : v)
	{
		cout << e << " ";
	}
	cout << endl;
	auto p = find(v.begin(), v.end(), 3));
	if (p != v.end())
	{
		v.insert(p, 30);
	}
	for (auto e : v)
	{
		cout << e << " ";
	}

}

原因
当我们插入30时会发生第二次扩容,此时find 找到了vector容器元素为3的位置并返回指向3的迭代器,但是因为vector扩容后会对这原来的vector空间销毁,此时返回的p并不是原来3的迭代器位置,此时p已经变成了野指针,insert时pos会导致导致程序崩溃。
以下insert 和 reserve 方法的底层实现:
在这里插入图片描述在这里插入图片描述
pos此时的值已经变成了随机值。
在这里插入图片描述
改善方法探讨:
我们可以在扩容后更新pos指向的元素的位置。但是此时又有一个问题,我们在扩容后重新修正了pos的位置时,却发现p依旧是失效的,原因是pos是形参,p是实参。形参的改变并不会影响到实参。
在这里插入图片描述
此时,我们可以在pos形参位置加入引用,但是当我们使用
v.begin()为实参传入时,却发现调用不成功,因为此时的v.begin()
返回的值具有常性,权限由小变大。
在这里插入图片描述

又或者我们可以返回pos位置的引用,又说明引用的值可以被修改,这又与STL insert方法的实现相违背。

解决方法
当使用insert后又要使用insert或者erase操作进行扩容或者缩容时,因为在insert时,扩容后已经重新对pos进行赋值,但是因为pos终究是形参的改变无法影响到实参,所以在insert后要使用变量去接受pos的位置。

void test_vector1()
{
	vector <int> v;
	v.push_back(1);
	v.push_back(2);
	v.push_back(3);
	v.push_back(4);
	for (auto e : v)
	{
		cout << e << " ";
	}
	cout << endl;
	auto p = find(v.begin(), v.end(), 3);
	if (p != v.end())
	{
		v.insert(p, 30);
		   //插入一个数后及时接受pos的位置后才能继进行插入。
		auto p = find(v.begin(), v.end(), 3);
		v.insert(p, 30);
	}
	for (auto e : v)
	{
		cout << e << " ";
	}

}
int main()
{
	test_vector1();
}

2:erase删除或者insert插入时元素移动迭代器问题:

如果erase函数具有缩容功能时,也会导致迭代器失效,所以erase之后我们尽量不要直接对它进行访问。
第一种情况
程序正常运行:
如果我们插入5个元素,当it指向元素5时,it正好指向v.end(),所以程序正常退出。
在这里插入图片描述
第二种情况
程序崩溃:
如果我们插个4个元素,当it指向元素4进行删除后 it++,但是v.end()指向了元素3的下一个位置,而此时it又进行++,与v.end()正好错过了。所以本来应该删除4之后结束循环的,如今只能进行删除。如果一直循环下去的化,it就变成了野指针的,在erase中对野指针访问后就会造成程序崩溃。
在这里插入图片描述
第三种情况
程序结果不对:
当我们连续插入偶数时,it指向元素2删除后,原来在2后面的4向前移动了以为,在下一次while循环中,原本it指向4的现在指向了元素3,元素4未被删除。
在这里插入图片描述
解决方法
针对插入4个元素时,在不对迭代器i重新赋值时,我们可以当it指向偶数就删除,指向奇数就++;这样在最后删除4后,it就不会像之前++了,此时it指向的位置正好等于v.end(),结束循环。
在这里插入图片描述

再例如:我们在偶数之前插入这个偶数的两倍。
如果我们不对it进行操作,直接让it++后进行访问,
又因为再2的前面插入一个数后,此时的2已经在原来的位置基础上向后移动了以为,这样it又会指向元素2,此刻便会不断循环对2的前面一个元素插入数字。
造成程序崩溃。
在这里插入图片描述
解决办法
我们要解决插入元素后it一直指向同一个数问题,如果it指向偶数,可以在偶数前插入一个术后,it++两次,从而跳过已经插过数的偶数了。如果为奇数,则直接++it。
在这里插入图片描述

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

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

相关文章

什么是“奥卡姆剃刀”,如何用“奥卡姆剃刀”解决复杂问题?复杂问题简单化

什么是“奥卡姆剃刀”&#xff0c;如何用“奥卡姆剃刀”解决复杂问题&#xff1f;复杂问题简单化问题什么是“奥卡姆剃刀”?如何使用“奥卡姆剃刀”解决问题复杂问题简单化“汉隆剃刀”小结问题 假设你在夜空中看到一颗闪闪发光的「不明飞行物」&#xff0c;你认为这会是什么呢…

优秀的IC/FPGA开源项目(六)-手语字母翻译器

《优秀的IC/FPGA开源项目》是新开的系列&#xff0c;旨在介绍单一项目&#xff0c;会比《优秀的 Verilog/FPGA开源项目》内容介绍更加详细&#xff0c;包括但不限于综合、上板测试等。两者相辅相成&#xff0c;互补互充~一种智能手套&#xff0c;可将手语字母翻译成带显示器的书…

【分享】订阅集简云畅捷通T+cloud连接器自动同步财务费用单至畅捷通

方案场景 伴随公司发展和数字化水平提高&#xff0c;大量的财务单据需要手动审核和录入&#xff0c;这些重复机械的操作占据大量人力&#xff0c;同时极容易出现数据出错或丢失等情况&#xff0c;严重影响着企业经营效率。 使用集简云提供服务的畅捷通TCloud钉钉连接器完成财…

UML 类关系(详解)——依赖、关联、聚合、组合、泛化

概述 在学习面向对象设计时&#xff0c;类关系涉及依赖、关联、聚合、组合和泛化&#xff08;继承&#xff09;这五种关系&#xff0c;耦合度依次递增。关于耦合度&#xff0c;可以简单地理解为当一个类发生变更时&#xff0c;对其他类造成的影响程度&#xff0c;影响越小则耦…

电子发票打印工具 v2023.02.27 免费的PDF发票打印软件

电子发票打印工具一款PDF发票打印辅助软件,因为单位有很多电子发票需要打印,每次打印都要用reader阅读器打开,选打印,选份数,选纸张,选纸盒,当然,有些发票有清单页的,以上步骤请再重复一遍。多张发票请重复N遍,忍了多年,终于决心开发一款方便用于打印PDF发票的辅助工…

源码numpy笔记

参考文章 numpy学习 numpy中的浅复制和深复制的详细用法 numpy中的np.where torch.gather() Numpy的核心数据结构&#xff0c;就叫做array就是数组&#xff0c;array对象可以是一维数组&#xff0c;也可以是多维数组 array本身的属性 shape&#xff1a;返回一个元组&#xf…

【数据库增删查改进阶版】保姆级教程带大家去学习更加复杂的sql语句,各种各样的约束以及各种各样的查询

前言&#xff1a; 大家好&#xff0c;我是良辰丫&#x1f345;&#x1f345;&#x1f345;&#xff0c;上一篇数据库我们一起学习了基础版本的增删查改&#xff0c;今天我们将接触更高级的增删查改&#xff0c;主要是学习一些约束条件&#xff0c;你们准备好了嘛&#xff1f;开…

华为OD机试题,用 Java 解【流水线】问题

最近更新的博客 华为OD机试题,用 Java 解【停车场车辆统计】问题华为OD机试题,用 Java 解【字符串变换最小字符串】问题华为OD机试题,用 Java 解【计算最大乘积】问题华为OD机试题,用 Java 解【DNA 序列】问题华为OD机试 - 组成最大数(Java) | 机试题算法思路 【2023】使…

LearnOpenGL-入门-8.坐标系统

本人刚学OpenGL不久且自学&#xff0c;文中定有代码、术语等错误&#xff0c;欢迎指正 我写的项目地址&#xff1a;https://github.com/liujianjie/LearnOpenGLProject LearnOpenGL中文官网&#xff1a;https://learnopengl-cn.github.io/ 文章目录坐标系统概述局部空间世界空…

干货收藏|医疗数据安全、临床业务容灾、智能运维及数字化转型方案集锦

数智赋能&#xff0c;助力医院高质量发展&#xff01;历时三天的2022中华医院信息网络大会&#xff08;CHINC&#xff09;圆满落下帷幕&#xff0c;美创科技赴五年之约&#xff0c;与医疗行业用户朋友在深圳再聚交流&#xff0c;也带来关于“医疗行业数据安全、数字化转型”的新…

【再临数据结构】Day1. 稀疏数组

前言 这不单单是稀疏数组的开始&#xff0c;也是我重学数据结构的开始。因此&#xff0c;在开始说稀疏数组的具体内容之前&#xff0c;我想先说一下作为一个有着十余年“学龄”的学生&#xff0c;所一直沿用的一个学习方法&#xff1a;3W法。我认为&#xff0c;只有掌握了正确的…

react的严格模式 和 解决react useEffect执行两次

useEffect执行两次 这个问题&#xff0c;主要是刚接触react的时候发的问题&#xff0c;当时也没总结。现在回过头来再总结一次&#xff01;&#xff01;&#xff01; 文章目录useEffect执行两次前言一、为什么useEffect执行两次1.React的严格模式&#xff08;模版创建项目&…

Hadoop综合案例 - 聊天软件数据

目录1、聊天软件数据分析案例需求2、基于Hive数仓实现需求开发2.1 建库2.2 建表2.3 加载数据2.4 ETL数据清洗2.5 需求指标统计---都很简单3、FineBI实现可视化报表3.1 FineBI介绍3.2 FineBI配置数据3.3 构建可视化报表1、聊天软件数据分析案例需求 MR速度慢—引入hive 背景&a…

深度剖析指针(中)——“C”

各位CSDN的uu们你们好呀&#xff0c;今天小雅兰的内容仍旧是深度剖析指针噢&#xff0c;在上一篇博客中&#xff0c;我已经写过了字符指针、数组指针、指针数组、数组传参和指针传参的知识点&#xff0c;那么这篇博客小雅兰会讲解一下函数指针、函数指针数组 、指向函数指针数组…

【Spark分布式内存计算框架——Spark Streaming】8. Direct 方式集成底层原理 集成Kafka 0.10.x

Direct 方式集成底层原理 SparkStreaming集成Kafka采用Direct方式消费数据&#xff0c;如下三个方面优势&#xff1a; 第一、简单的并行度&#xff08;Simplified Parallelism&#xff09; 读取topics的总的分区数目 每批次RDD中分区数目&#xff1b;topic中每个分区数据 被…

布局三八女王节,巧借小红书数据分析工具成功引爆618

对于小红书“她”经济来说&#xff0c;没有比三八节更好的阵地了。伴随三八女王节逐渐临近&#xff0c;各大品牌蓄势待发&#xff0c;这场开春后第一个S级大促活动&#xff0c;看看品牌方们可以做什么&#xff1f; 洞察流量&#xff0c;把握节点营销时机 搜索小红书2023年的三…

学员作品|微博“绿洲”APP产品分析

一产品架构1. 产品功能架构图绿洲的主要功能模块可以拆分为六部分&#xff1a;首页、发现、发布动态、个人中心、水滴、消息。整体功能架构图如下&#xff1a;2. 用户使用路径图用于浏览动态&#xff1a;用于发布动态&#xff1a;新用户引导路径&#xff1a;二市场分析1. 产品定…

【Linux】Linux中gcc/g++的使用

本期主题&#xff1a;程序的编译过程和gcc/g的使用博客主页&#xff1a;小峰同学分享小编的在Linux中学习到的知识和遇到的问题小编的能力有限&#xff0c;出现错误希望大家不吝赐&#x1f341; 1.背景知识 预处理&#xff08;进行宏替换&#xff0c;去注释&#xff0c;头文件的…

索引、索引失效、索引的存储

文章目录1.索引种类2.创建索引注意点3.索引失效的情况4.为什么索引使用B树存储&#xff08;1&#xff09;如果使用数组来存储索引&#xff08;2&#xff09;如果使用二叉查找树来存储索引&#xff08;3&#xff09;如果使用平衡二叉查找树来存储索引&#xff08;4&#xff09;如…

同为(TOWE)防雷产品助力福建移动南平分公司防雷改造

01 公司简介中国移动通信集团福建有限公司南平分公司属于福建移动地级分公司&#xff0c;所属行业为电信、广播电视和卫星传输服务。现已建成覆盖范围广、业务品种多、通信质量高的综合通信网络&#xff0c;具备行业领先的经营管理制度。移动通信大楼的综合防雷及地接系统&…