C++:vector基础讲解

news2024/10/5 18:25:59

hello,各位小伙伴,本篇文章跟大家一起学习《C++:vector基础讲解》,感谢大家对我上一篇的支持,如有什么问题,还请多多指教 !
如果本篇文章对你有帮助,还请各位点点赞!!!
在这里插入图片描述

话不多说,开始进入正题

文章目录

    • :maple_leaf:关于vector
    • :maple_leaf:vector构造函数
      • :leaves:vector() 无参构造(默认构造)
      • :leaves:vector(size_type n, const value_type& val = value_type()) 构造并初始化n个val
      • :leaves:vector(const vector& x) 拷贝构造
      • :leaves:vector (InputIterator first, InputIterator last);使用迭代器进行初始化构造
      • :leaves:vector{type n1,type n2,type n3...}
    • :maple_leaf:vector iterator 的使用
      • :leaves:begin
      • :leaves:end
    • :maple_leaf:vector关于空间的函数
      • :leaves:size
      • :leaves:capacity
      • :leaves:empty
      • :leaves:resize
      • :leaves:reserve
    • :maple_leaf:vector 增删查改
      • :leaves:push_back 尾插
      • :leaves:pop_back 尾删
      • :leaves:find 查找
      • :leaves:insert 插入
      • :leaves:erase 删除数据
      • :leaves:swap 交换vector的数据空间
      • :leaves:operator[] `[]`运算符重载
    • :maple_leaf:vector迭代器失效问题
      • :leaves: 会引起其底层空间改变的操作,都有可能是迭代器失效
      • :leaves:2. 指定位置元素的删除操作--erase
      • :leaves:注意:Linux下,g++编译器对迭代器失效的检测并不是非常严格,处理也没有vs下极端。
      • :leaves:与vector类似,string在插入+扩容操作+erase之后,迭代器也会失效
      • :leaves:解决办法

🍁关于vector

vector是表示可变大小数组的序列容器。能够用下标访问数组中的元素,相对于普通数组能够动态增长。本质讲,vector使用动态分配数组来存储它的元素。当新元素插入时候,这个数组需要被重新分配大小为了增加存储空间。其做法是,分配一个新的数组,然后将全部元素移到这个数组。就时间而言,这是一个相对代价高的任务,因为每当一个新的元素加入到容器的时候,vector并不会每次都重新分配大小。vector相比于其他容器访问元素会更加高效,但是对于不在末尾在插入数据和删除数据时效率更低。

🍁vector构造函数

🍃vector() 无参构造(默认构造)

vector<int>p;

🍃vector(size_type n, const value_type& val = value_type()) 构造并初始化n个val

vector<int>p1(10,1);
vector<string>p2(10,"abcd");

🍃vector(const vector& x) 拷贝构造

vector<int>p1(10,1);
vector<int>p2(p1);

🍃vector (InputIterator first, InputIterator last);使用迭代器进行初始化构造

string s1("abcde");
vector<char>p1(s1.begin(),s1.end());
vector<int>p2(p1.begin(),p1.end());

🍃vector{type n1,type n2,type n3…}

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

这种初始化,会生成一个类型为initializer_list的初始化列表,用来初始化v
在这里插入图片描述

这种初始化列表template< class T > class initializer_list;
是一种模板

🍁vector iterator 的使用

🍃begin

获取第一个数据位置的iterator/const_iterator

vector<int>p(10,1);
iterator lt = p.begin();
cout<<(*lt)<<endl;

🍃end

获取最后一个数据的下一个位置的iterator/const_iterator

vector<int>p(10,1);
iterator lt = p.end();

🍁vector关于空间的函数

🍃size

获取数据个数

vector<int>p(10,1);
cout<<p.size();

🍃capacity

获取容量大小

vector<int>p(10,1);
cout<<p.capacity();

🍃empty

判断是否为空

vector<int>p1(10,1);
cout<<p1.empty()<<endl;

vector<int>p2;
cout<<p2.empty()<<endl;

🍃resize

void resize (size_type n, value_type val = value_type());

改变vector的size,当扩容时n > 扩容前的capacity,也会影响capacity

缩容:

vector<int>p1(10,1);
cout<<p1.size()<<endl;
p1.resize(5)
vector<int>p1;
cout<<p1.size()<<endl;

扩容:

vector<int>p1(10,1);
cout<<p1.size()<<endl;
p1.resize(202)
vector<int>p1;
cout<<p1.size()<<endl;
for(auto& e : p1)
{
	cout<<e<<" ";
}
cout<<endl;

扩容时,可以传第二个参数,用此值来初始化扩容的空间,也可以不传第二个参数,会自动调用构造函数初始化扩容的空间

🍃reserve

改变vector的capacity

vector<int>p1(10,1);
cout<<p1.capacity()<<endl;
p1.reserve(100);
vector<int>p1;
cout<<p1.capacity()<<endl;

有些编译器会缩容,有些不会,一般不建议用reserve来缩容。
reserve只负责开辟空间,如果确定知道需要用多少空间,reserve可以缓解vector增容的代价缺陷问
题。

🍁vector 增删查改

🍃push_back 尾插

void push_back (const value_type& val);

vector<char>str(5,'a');
str.push_back('b');
cout << str[str.size() - 1];
cout << endl;

🍃pop_back 尾删

void pop_back();

vector<char>str(5,'a');
str.push_back('c');
cout << str[5] << endl;
cout << str.size() << endl;
str.pop_back();
cout << str.size() << endl;

🍃find 查找

(注意这个是算法模块实现,不是vector的成员接口)
template <class InputIterator, class T>
InputIterator find (InputIterator first, InputIterator last, const T& val);

vector<int>nums(10);
for(int i = 0;i < 10;++i)
{
	nums[i] = i;
}

🍃insert 插入

在指定位置前插入
(1)iterator insert (iterator position, const value_type& val);插入一个元素
(2)void insert (iterator position, size_type n, const value_type& val);插入n个元素
(3)template< class InputIterator >迭代器模板
void insert (iterator position, InputIterator first, InputIterator last);插入一段区间

可知返回值是迭代器类型,insert返回的是在插入元素的位置

int arr[3] = { 9,9,9 };
vector<int>v(3, 1);

v.insert(v.begin(), 1);// 一个元素
for (auto& e : v)
{
	cout << e << " ";
}
cout << endl;

v.insert(v.begin(), 10, 0);// n个元素
for (auto& e : v)
{
	cout << e << " ";
}
cout << endl;

v.insert(v.begin(), arr, arr + 3);// 一段区间
for (auto& e : v)
{
	cout << e << " ";
}
cout << endl;

cout << *lt << endl;

🍃erase 删除数据

删除position位置的数据
iterator erase (iterator position);iterator erase (iterator first, iterator last);

erase也是返回迭代器类型,返回的是删除数据的位置(pos)的后一个位置

int arr[3] = { 3,2,1 };
vector<int>v;
v.insert(v.begin(), arr, arr + 3);
for (auto& e : v)
{
	cout << e << " ";
}
cout << endl;
vector<int>::iterator lt = v.erase(v.begin());
for (auto& e : v)
{
	cout << e << " ";
}
cout << endl;
cout << *lt << endl;

🍃swap 交换vector的数据空间

在这里插入图片描述

void swap (vector& x);

int arr1[3] = { 3,2,1 };
int arr2[3] = { 1,2,3 };
vector<int>v1;
vector<int>v2;
v1.insert(v1.begin(), arr1, arr1 + 3);
v2.insert(v2.begin(), arr2, arr2 + 3);
for (auto& e : v1)// 交换前
{
	cout << e << " ";
}
cout << endl;
for (auto& e : v2)
{
	cout << e << " ";
}
cout << endl;

swap(v1, v2);// 交换后
for (auto& e : v1)
{
	cout << e << " ";
}
cout << endl;
for (auto& e : v2)
{
	cout << e << " ";
}
cout << endl;

🍃operator[] []运算符重载

reference operator[] (size_type n); 非const修饰
const_reference operator[] (size_type n) const; const修饰

通过下标访问vector的元素,很方便,像数组一样访问

int arr1[3] = { 3,2,1 };
vector<int>v1;
v1.insert(v1.begin(), arr1, arr1 + 3);
cout << v1[0] << " " << v1[1] << " " << v1[2] << endl;

🍁vector迭代器失效问题

迭代器的主要作用就是让算法能够不用关心底层数据结构,其底层实际就是一个指针,或者是对指针进行了封装,比如:vector的迭代器就是原生态指针T* 。因此迭代器失效,实际就是迭代器底层对应指针所指向的空间被销毁了,而使用一块已经被释放的空间,造成的后果是程序崩溃(即如果继续使用已经失效的迭代器,程序可能会崩溃)。

对于vector可能会导致其迭代器失效的操作有:

🍃 会引起其底层空间改变的操作,都有可能是迭代器失效

比如:resize、reserve、insert、assign、push_back 等。

#include <iostream>
using namespace std;
#include <vector>
 
int main()
{
 vector<int> v{1,2,3,4,5,6};
 
 auto it = v.begin();
 
 // 将有效元素个数增加到100个,多出的位置使用8填充,操作期间底层会扩容
 // v.resize(100, 8);
 
 // reserve的作用就是改变扩容大小但不改变有效元素个数,操作期间可能会引起底层容量改变
 // v.reserve(100);
 
 // 插入元素期间,可能会引起扩容,而导致原空间被释放
 // v.insert(v.begin(), 0);
 // v.push_back(8);
 
 // 给vector重新赋值,可能会引起底层容量改变
 v.assign(100, 8);
 
 /*
 出错原因:以上操作,都有可能会导致vector扩容,也就是说vector底层原理旧空间被释放掉,
而在打印时,it还使用的是释放之间的旧空间,在对it迭代器操作时,实际操作的是一块已经被释放的
空间,而引起代码运行时崩溃。
 解决方式:在以上操作完成之后,如果想要继续通过迭代器操作vector中的元素,只需给it重新
赋值即可。
 */
 while(it != v.end())
 {
 cout<< *it << " " ;
 ++it;
 }
 cout<<endl;
 return 0;
}

🍃2. 指定位置元素的删除操作–erase

#include <iostream>
using namespace std;
#include <vector>
 
int main()
{
 int a[] = { 1, 2, 3, 4 };
 vector<int> v(a, a + sizeof(a) / sizeof(int));
 
 // 使用find查找3所在位置的iterator
 vector<int>::iterator pos = find(v.begin(), v.end(), 3);
 
 // 删除pos位置的数据,导致pos迭代器失效。
 v.erase(pos);
 cout << *pos << endl; // 此处会导致非法访问
 return 0;
}

通常情况下earse删除数据后,pos位置后的元素会向前移动,迭代器不会失效,但如果删除的是最后一个元素,就会发生迭代器失效,因为此时迭代器指向end位置,end位置是没有元素的。

🍃注意:Linux下,g++编译器对迭代器失效的检测并不是非常严格,处理也没有vs下极端。

// 1. 扩容之后,迭代器已经失效了,程序虽然可以运行,但是运行结果已经不对了
int main()
{
 vector<int> v{1,2,3,4,5};
 for(size_t i = 0; i < v.size(); ++i)
 cout << v[i] << " ";
 cout << endl;
 
 auto it = v.begin();
 cout << "扩容之前,vector的容量为: " << v.capacity() << endl;
 // 通过reserve将底层空间设置为100,目的是为了让vector的迭代器失效 
 v.reserve(100);
 cout << "扩容之后,vector的容量为: " << v.capacity() << endl;
 
 // 经过上述reserve之后,it迭代器肯定会失效,在vs下程序就直接崩溃了,但是linux下不会
 // 虽然可能运行,但是输出的结果是不对的
 while(it != v.end())
 {
 cout << *it << " ";
 ++it;
 }
 cout << endl;
 return 0;
}
 
程序输出:
1 2 3 4 5 
扩容之前,vector的容量为: 5
扩容之后,vector的容量为: 100
0 2 3 4 5 409 1 2 3 4 5
// 2. erase删除任意位置代码后,linux下迭代器并没有失效
// 因为空间还是原来的空间,后序元素往前搬移了,it的位置还是有效的
#include <vector>
#include <algorithm>
 
int main()
{
 vector<int> v{1,2,3,4,5};
 vector<int>::iterator it = find(v.begin(), v.end(), 3);
 v.erase(it);
  
 cout << *it << endl;
 while(it != v.end())
 {
 cout << *it << " ";
 ++it;
 }
 cout << endl;
 return 0;
}
 
程序可以正常运行,并打印:
4
4 5
// 3: erase删除的迭代器如果是最后一个元素,删除之后it已经超过end
// 此时迭代器是无效的,++it导致程序崩溃
int main()
{
 vector<int> v{1,2,3,4,5};
 // vector<int> v{1,2,3,4,5,6};
 auto it = v.begin();
 while(it != v.end())
 {
 if(*it % 2 == 0)
 v.erase(it);
 ++it;
 }
 
 for(auto e : v)
 cout << e << " ";
 cout << endl;
 return 0;
}
 
========================================================
// 使用第一组数据时,程序可以运行
[sly@VM-0-3-centos 20220114]$ g++ testVector.cpp -std=c++11
[sly@VM-0-3-centos 20220114]$ ./a.out
1 3 5 
=========================================================
// 使用第二组数据时,程序最终会崩溃
[sly@VM-0-3-centos 20220114]$ vim testVector.cpp 
[sly@VM-0-3-centos 20220114]$ g++ testVector.cpp -std=c++11
[sly@VM-0-3-centos 20220114]$ ./a.out
Segmentation fault

🍃与vector类似,string在插入+扩容操作+erase之后,迭代器也会失效

string s1("hello");
string::iterator lt = --s1.end();
for (auto& e : s1)
{
	cout << e << " ";
}
cout << endl;
s1.erase(--s1.end());// erase后,lt迭代器已经失效
for (auto& e : s1)
{
	cout << e << " ";
}
cout << *lt << endl;// 会运行崩溃

🍃解决办法

迭代器失效解决办法:在使用前,对迭代器重新赋值即可。
你学会了吗?
好啦,本章对于《C++:vector基础讲解》的学习就先到这里,如果有什么问题,还请指教指教,希望本篇文章能够对你有所帮助,我们下一篇见!!!

如你喜欢,点点赞就是对我的支持,感谢感谢!!!

请添加图片描述

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

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

相关文章

网络编程day7

思维导图 数据库编程实现学生管理系统 #include <header.h> #define ID 1 #define NAME 2 #define AGE 3 #define SCORE 4 int do_add(sqlite3 *ppdb) {int add_numb;char add_name[20];int add_age;double add_score;printf("enter student id:");scanf(&quo…

1076: 判断给定有向图是否存在回路

解法&#xff1a; 直观的方法用邻接矩阵dfs,这是错误的代码 #include<iostream> #include<vector> using namespace std; int arr[100][100]; int f 0; void dfs(vector<int>& a, int u) {a[u] 1;for (int i 0; i < a.size(); i) {if (arr[u][i]…

绝缘监测系统在1kV 及以下低压配电系统的应用

安科瑞电气股份有限公司 祁洁 acrelqj 一、系统概述 Acrel-2000L/A 绝缘监测系统设备适用于 1kV 及以下低压配电系统。该设备可以集中采集监测显示绝缘监测仪的数据&#xff0c;实现最多 8 个绝缘监测仪的数据&#xff0c;并且实时记录告警信息和曲线查询。匹配的绝缘监测仪…

bootstrap入门

官方网站&#xff1a;全局 CSS 样式 Bootstrap v3 中文文档 | Bootstrap 中文网 里面各种可以直接用的组件 不全的话可以网上搜索Boostrap常用的按钮样式_btn-large-dim-CSDN博客 怎么在vue项目中使用呢 npm install bootstrap 下载下来然后在main.js加上红框三句后&#…

SpringCloudAlibaba:6.2RocketMQ的普通消息的使用

简介 普通消息也叫并发消息&#xff0c;是发送效率最高&#xff0c;使用最多的一种 依赖 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSch…

1701java药品进销存管理系统Myeclipse开发sqlserver数据库web结构java编程计算机网页项目

一、源码特点 java web药品进销存管理系统是一套完善的java web信息管理系统&#xff0c;对理解JSP java编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用B/S模式开发。开发环境 为 TOMCAT7.0,Myeclipse8.5开发&#xff0c;数据库为s…

解决vue3 vite打包报Root file specified for compilation问题

解决方法&#xff1a; 修改package.json打包命令 把 "build": "vue-tsc --noEmit && vite build" 修改为 "build": "vite build" 就可以了 另外关于allowJs这个问题&#xff0c;在tsconfig.json文件中配置"allowJs&qu…

重学java 39.多线程 — 线程安全

逐渐成为一个情绪稳定且安静成长的人 ——24.5.24 线程安全 什么时候发生&#xff1f; 当多个线程访问同一个资源时&#xff0c;导致了数据有问题&#xff0c;出现并发问题&#xff0c;数据不能及时更新&#xff0c;导致数据发生错误&#xff0c;出现线程安全问题 多线程安全问…

【不太正常的题】LeetCode.232:用栈的函数接口实现队列

&#x1f381;个人主页&#xff1a;我们的五年 &#x1f50d;系列专栏&#xff1a;初阶数据结构刷题 &#x1f389;欢迎大家点赞&#x1f44d;评论&#x1f4dd;收藏⭐文章 &#x1f697; 1.问题描述&#xff1a; 题目中说了只能使用两个栈实现队列&#xff0c;并且只能使用…

【Crypto】password

文章目录 password解题感悟 password 试试flag{zs19900315} 提交成功 解题感悟 这题有点大病

软考--软件设计师--试题六--工厂方法模式(Factory Method)

工厂方法模式(Factory Method) 1、意图 定义一个用于创建对象的接口&#xff0c;让子类决定实例化哪儿一个类&#xff0c;factory method使一个类的实例化延迟到其子类。 2、结构 3、适用性 a、当一个类不知道它所必须创建的对象的类的时候。 b、当一个类希望由它的子类来指定…

UPPAAL使用方法

UPPAAL使用方法 由于刚开始学习时间自动机及其使用方法&#xff0c;对UPPAAL使用不太熟悉&#xff0c;网上能找到的教程很少&#xff0c;摸索了很久终于成功实现一个小例子&#xff0c;所以记录一下详细教程。 这里用到的例子参考【UPPAAL学习笔记】1&#xff1a;基本使用示例…

CyberScheduler调度引擎

CyberScheduler 架构设计 1. 多租户架构&#xff0c;支持 SaaS 化部署和私有化部署 2. 多源异构数据&#xff08;多种集群、数据库&#xff09;、多计算引擎、多类型任务的统一编排调度 3. 灵活资源管理能力&#xff0c;支持不同类型任务的资源管理和资源隔离&#xff0c;优…

docker 挂载运行镜像

文章目录 前言docker 挂载运行镜像1. 作用2. 命令3. 测试 前言 如果您觉得有用的话&#xff0c;记得给博主点个赞&#xff0c;评论&#xff0c;收藏一键三连啊&#xff0c;写作不易啊^ _ ^。   而且听说点赞的人每天的运气都不会太差&#xff0c;实在白嫖的话&#xff0c;那欢…

1077: 平衡二叉树的判定

解法&#xff1a; 平衡二叉树是一种特殊的二叉树&#xff0c;它满足以下两个条件&#xff1a; 左子树和右子树的高度差不超过1&#xff08;即&#xff0c;左右子树高度差的绝对值不超过1&#xff09;。左子树和右子树都是平衡二叉树。 后序遍历过程中每次判断左右子树高度差…

收入极高的副业兼职,单价179元的养生爆款卖出3000份

做养生赛道切忌不要只靠接广告变现&#xff0c;换个思路以产品为核心&#xff0c;走低粉高变现才是变现效率最高的。 周周近财&#xff1a;让网络小白少花冤枉钱&#xff0c;赚取第一桶金 今天&#xff0c;我们要分析的这位养生博主&#xff0c;仅凭一款售价为179元的温通膏&a…

STM32 MAP文件结合固件文件分析

文章目录 加载域的结束地址并不是固件的结束地址&#xff1f;ROM中执行域的描述RAM中执行域的描述问题分析 中断向量表在固件中的存储位置代码段在固件中的位置只读数据Regin$$Table RW Data段其中的内部机理 总结 MAP 文件分析可以参考之前的文章 程序代码在未运行时在存储器…

漫谈企业信息化安全 - 勒索软件攻击

一、引言 首先&#xff0c;网络攻击是一个非常广泛的话题&#xff0c;网络攻击从一般分类上包含了恶意软件攻击、钓鱼攻击、拒绝服务攻击&#xff08;DoS/DDoS&#xff09;、中间人攻击、SQL注入、跨站脚本、0-Day攻击、供应链攻击、密码攻击等等&#xff0c;勒索软件攻击只是…

EfficientSAM分割对象后求其中图像中的高

1 分割对象 EfficientSAM https://github.com/yformer/EfficientSAM 2 计算在图像中最高点即y值最小点 import os import cv2def read_images(folder_path):image_files [f for f in os.listdir(folder_path) iff.endswith(".jpg") or f.endswith(".png&quo…

OSPF状态机及网络接口类型

、OSPF 状态机 Down一旦接收到hello 包进人下一个状态机 Init 初始化接收到的hello 包中&#xff0c;若存在本地的 RID&#xff0c;进入下一状态 2way 双向通讯--邻居关系建立的标志 条件匹配:点到点网络直接进入下一个状态机 MA 网络将进行 DR/BDR 选举(40S) 非 DR…