STL之vector

vector简单介绍

  1. vector是一个STL的一个容器,行为类似于变成数组,也就是存储空间是连续的,同时其空间大小又是动态可变的。

  2. vector访问元素的效率很高可以通过下标直接访问,但是其占用的空间很多,插入删除元素的效率很低。

  3. 使用vector时记得包含头文件

    c++ 复制代码
    #include <vector>

vector的构造函数

vector本质是类模板,可以存储任何类型的数据。数组在声明前需要加上数据类型,而vector则通过模板参量设定类型。比如,声明一个int型的vector数组。

c++ 复制代码
vector<int> arr1;								//一个空数组
vector<int> arr2 {1, 2, 3, 4, 5};				//包含1、2、3、4、5五个变量
vector<int> arr3(4);							//开辟4个空间,值默认为0
vector<int> arr4(5, 3);							//5个值为3的数组
vector<int> arr5(arr4);							//将arr4的所有值复制进去,和arr4一样
vector<int> arr6(arr4.begin(), arr4.end());		//将arr4的值从头开始到尾复制
vector<int> arr7(arr4.rbegin(), arr4.rend());	//将arr4的值从尾到头复

也就是说想要创建一个vector可以通过

  1. 初始化列表构造如arr2

  2. 含有n个元素的vetcor如arr3

  3. 含有n元素且值为value如arr4

  4. 拷贝构造如arr5

  5. 通过迭代器区间构造如arr6和arr7

vector的容量

这里主要介绍其中的resize和reserve。

vector的容量包括:size,capacity,empty,resize和reserve。

  • size顾名思义就是得到数据的个数
  • capacity则是得到容量大小
  • empty是判断是否为空
  • resize是改变size,reserve是改变capacity这也是两者最根本的区别

reserve主要是用来预留空间在已知需要多少空间的情况下避免不必要的扩容

c++ 复制代码
void resize(size_type n, value_type val = value_type());

- 如果n<当前容器的size,则将元素减少到前n个,移除多余的元素(并销毁)
- 如果n>当前容器的size,则在容器中追加元素,如果val指定了,则追加的元素为val的拷贝,否则,默认初始化
- 如果n>当前容器容量,内存会自动重新分配


void reserve(size_type n)

- 如果n>容器的当前capacity,该函数会使得容器重新分配内存使capacity达到n
- 任何其他情况,该函数调用不会导致内存重新分配,并且容器的capacity不会改变
- 该函数不会影响向量的size而且不会改变任何元素

resize是改变size并且创建对象

c++ 复制代码
1、resize(n) 

	调整容器的长度大小,使其能容纳n个元素。
	
	如果n小于容器的当前的size,则删除多出来的元素。
	
	否则,添加采用值初始化的元素。

2、 resize(n,t)

	多一个参数t,将所有新添加的元素初始化为t。

vector的元素访问以及遍历

  1. 通过[]下标访问

    c++ 复制代码
    for (int i = 0; i < arr.size(); i++)
    {
    	cout << arr[i] << endl;
  2. 范围for

c++ 复制代码
for (auto num : arr)
{
	cout << num << endl;
}

3.迭代器访问

  • 通过迭代器访问从begin()到end(),需要定义iterator,当然可以用auto替代。
  • begin()表示第一个元素,而end()不是最后一个元素,end()是最后一个元素的前一个位置
  • rebegin()返回的是最后一个元素,reend()则返回的是第一个元素的前一个元素
c++ 复制代码
//迭代器:vector<int>::iterator
for (vector<int>::iterator it = arr.begin(); it != arr.end(); it++)
{
    cout << *it << endl;
}
//迭代器:vector<int>::reverse_iterator
for (vector<int>::reverse_iterator it = arr.rbegin(); it != arr.rend(); it++)
{
    cout << *it << endl;
}

vector使用时的注意事项

1.迭代器失效

vector的每一次扩容都不是在原地扩容,而是新开辟一块儿空间后将原先的数据拷贝到新空间

当我们使用insert()插入新的元素时

c++ 复制代码
vector<int> arr;
arr.push_back(1);
arr.push_back(2);
arr.push_back(3);
arr.push_back(4);
arr.push_back(5);
auto pos = find(arr.begin(),arr.end(),3);
arr.insert(pos,30);
arr.insert(pos,40);

此时就会出现问题

在使用erase()时也会有问题

c++ 复制代码
vector<int> v;
v.push_back(1);
v.push_back(2);
v.push_back(3);
v.push_back(4);
v.push_back(4);
v.push_back(6);

for (auto e : v)
{
	cout << e << " ";
}
cout << endl;
auto it = v.begin();
while (it != v.end())
{
	if (*it % 2 == 0)
	{
		it = v.erase(it);
	}
	++it;
}

for (auto e : v)
{
	cout << e << " ";
}
cout << endl;

这段代码想要删除所有的偶数

可是在使用erase()时它返回的是被删除元素的下一个位置的迭代器

那么解决方法也很简单

  1. 对于insert来说

在pos位置使用一次insert后

不要再次直接访问pos迭代器

一定要更新了pos之后再去访问!

  1. 对于erase来说

删除后不用再++迭代器

只用在没删除的时候再++

2深浅拷贝问题

拷贝vector时不要使用memcpy这是一种浅拷贝即只是拷贝了相同的数据并没有开辟新的空间浅拷贝是按字节拷贝,会导致拷贝后迭代器指向同一块空间。

对于insert来说`**

在pos位置使用一次insert后

不要再次直接访问pos迭代器

一定要更新了pos之后再去访问!

  1. 对于erase来说

删除后不用再++迭代器

只用在没删除的时候再++

相关推荐
艾莉丝努力练剑1 小时前
【LeetCode&数据结构】单链表的应用——反转链表问题、链表的中间节点问题详解
c语言·开发语言·数据结构·学习·算法·leetcode·链表
还债大湿兄3 小时前
《C++内存泄漏8大战场:Qt/MFC实战详解 + 面试高频陷阱破解》
c++·qt·mfc
深圳卢先生5 小时前
CentOS 安装jenkins笔记
笔记·centos·jenkins
倔强青铜35 小时前
苦练Python第18天:Python异常处理锦囊
开发语言·python
u_topian5 小时前
【个人笔记】Qt使用的一些易错问题
开发语言·笔记·qt
珊瑚里的鱼6 小时前
LeetCode 692题解 | 前K个高频单词
开发语言·c++·算法·leetcode·职场和发展·学习方法
AI+程序员在路上6 小时前
QTextCodec的功能及其在Qt5及Qt6中的演变
开发语言·c++·qt
xingshanchang6 小时前
Matlab的命令行窗口内容的记录-利用diary记录日志/保存命令窗口输出
开发语言·matlab
Risehuxyc6 小时前
C++卸载了会影响电脑正常使用吗?解析C++运行库的作用与卸载后果
开发语言·c++
AI视觉网奇6 小时前
git 访问 github
运维·开发语言·docker