深入了解vector

news/2024/11/14 19:17:23/

vector

    • 1. vector的介绍及使用
      • 1.1 vector的介绍
      • 1.2 vector的使用
        • 1.2.1 vector的定义((constructor)构造函数声明)
        • 1.2.2 vector iterator 的使用
        • 1.2.3 vector Capacity
        • 1.2.4 vector Modifiers
        • 1.2.4 vector 迭代器失效问题
    • 2. vector模拟实现

1. vector的介绍及使用

1.1 vector的介绍

在C++中,vector是一个模板类,它是一个多功能的、能够操作多种数据结构和算法的函数库。它是C++标准模板库中的部分内容,用于表示动态数组。vector可以自动扩展以容纳新元素,并且可以在任何位置插入或删除元素。

vector的优点:

  • vector是一个动态数组,可以自动扩展以容纳新元素。
  • vector支持随机访问,可以通过下标访问元素。
  • vector支持在任何位置插入或删除元素。

vector的缺点:

  • vector的插入和删除操作可能会导致内存重新分配,从而导致性能下降。
  • vector只能存储相同类型的元素。

1.2 vector的使用

1.2.1 vector的定义((constructor)构造函数声明)

default (1) vector (); //无参构造
fill (2) explicit vector (size_type n, const value_type& val = value_type()) // 构造并初始化n个val
range (3) vector (InputIterator first, InputIterator last); //使用迭代器进行初始化构造
copy (4) vector (const vector& x); //拷贝构造

1.2.2 vector iterator 的使用

在这里插入图片描述

vector的迭代器是一种指针,它可以遍历vector中的元素。vector的迭代器支持随机访问,可以通过下标访问元素。反向迭代器是一种特殊的迭代器,它可以从后向前遍历容器中的元素。
反向迭代器和正向迭代器的区别在于:对正向迭代器进行++操作时,迭代器会指向容器中的后一个元素;而对反向迭代器进行++操作时,迭代器会指向容器中的前一个元素。

const_iterator是一种只读迭代器,它不能修改vector中的元素。只读迭代器可以用于遍历vector中的元素,但不能用于修改vector中的元素。

#include <iostream>
#include <vector>
using namespace std;int main()
{vector<int> v = { 1, 2, 3, 4, 5 };// 使用begin和end遍历vector中的元素for (auto i = v.begin(); i != v.end(); ++i)cout << *i << " ";cout << endl;// 使用rbegin和rend遍历vector中的元素for (auto i = v.rbegin(); i != v.rend(); ++i)cout << *i << " ";cout << endl;return 0;
}

在这里插入图片描述

1.2.3 vector Capacity

在这里插入图片描述
用如下代码查看vector 空间增长问题
VS平台下:

#include <iostream>
#include <vector>
using namespace std;int main()
{vector<int> v;cout << "capacity: " << v.capacity() << endl;while (1){v.push_back(1);if (v.size() == v.capacity()){cout << "capacity: " << v.capacity() << endl;}}return 0;
}

在这里插入图片描述
gcc平台下:
在这里插入图片描述
capacity的代码在vs和g++下分别运行会发现,vs下capacity是按1.5倍增长的,g++是按2倍增长的。

1.2.4 vector Modifiers

在这里插入图片描述
assign()是vector的成员函数之一,它可以用于将新元素分配给vector。assign()函数有多个重载版本,可以接受不同类型的参数。下面是一些常见的用法:

// assign()函数的第一个版本
vector<int> v1;
v1.assign(5, 10); // 将5个值为10的元素分配给v1// assign()函数的第二个版本
vector<int> v2;
int arr[] = {1, 2, 3, 4, 5};
v2.assign(arr, arr + 5); // 将数组arr中的元素分配给v2// assign()函数的第三个版本
vector<int> v3;
vector<int> v4;
v3.assign(v4.begin(), v4.end()); // 将v4中的元素分配给v3

emplace()是vector的成员函数之一,它可以用于在vector中构造新元素。emplace()函数有多个重载版本,可以接受不同类型的参数。下面是一些常见的用法:

// emplace()函数的第一个版本
vector<pair<int, string>> v1;
v1.emplace_back(1, "hello"); // 在v1中构造一个pair<int, string>类型的元素// emplace()函数的第二个版本
vector<complex<double>> v2;
v2.emplace(v2.begin(), 1.0, 2.0); // 在v2的开头构造一个complex<double>类型的元素// emplace()函数的第三个版本
vector<string> v3;
v3.emplace_back("hello"); // 在v3中构造一个string类型的元素在这里插入代码片

1.2.4 vector 迭代器失效问题

vector的迭代器失效问题是指,当vector中的元素被添加、删除或插入时,迭代器可能会失效。如果一个迭代器失效了,那么它就不能再用于访问vector中的元素。

下面是一些常见的导致迭代器失效的操作:

  1. 会引起其底层空间改变的操作,都有可能是迭代器失效,比如:resize、reserve、insert、assign、
    push_back等。
  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所在位置的iteratorvector<int>::iterator pos = find(v.begin(), v.end(), 3);// 删除pos位置的数据,导致pos迭代器失效。v.erase(pos);cout << *pos << endl; // 此处会导致非法访问return 0;
}

在这里插入图片描述

erase删除pos位置元素后,pos位置之后的元素会往前搬移,没有导致底层空间的改变,理论上讲迭代器不应该会失效,但是:如果pos刚好是最后一个元素,删完之后pos刚好是end的位置,而end位置是没有元素的,那么pos就失效了。因此删除vector中任意位置上元素时,vs就认为该位置迭代器失效
了。
迭代器失效解决办法:在使用前,对迭代器重新赋值即可。

#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所在位置的iteratorvector<int>::iterator pos = find(v.begin(), v.end(), 3);// 删除pos位置的数据,导致pos迭代器失效。pos = v.erase(pos); // 按照下面方式写,运行时程序会崩溃,因为erase(pos)之后// pos位置的迭代器就失效了// s.erase(pos);cout << *pos << endl; // 此处会导致非法访问return 0;
}

在这里插入图片描述

2. vector模拟实现

在这里插入图片描述
在stl源码中,可以看到vector使用三个iterator来实现的。

template<class T>class vector{public:typedef T* iterator;typedef const T* const_iterator;vector(){}//vector<int> v(10, 5);vector(size_t n, const T& val = T()){reserve(n);for (size_t i = 0; i < n; ++i){push_back(val);}}vector(int n, const T& val = T()){reserve(n);for (int i = 0; i < n; ++i){push_back(val);}}vector(const vector<T>& v){_start = new T[v.capacity()];//必须用深拷贝for (size_t i = 0; i < v.size(); ++i){_start[i] = v._start[i];}_finish = _start + v.size();_end_of_storage = _start + v.capacity();}// [first, last)template <class InputIterator>vector(InputIterator first, InputIterator last){while (first != last){push_back(*first);++first;}}~vector(){delete[] _start;_start = _finish = _end_of_storage = nullptr;}iterator begin(){return _start;}iterator end(){return _finish;}const_iterator begin() const{return _start;}const_iterator end() const{return _finish;}void resize(size_t n, T val = T()){if (n < size()){// 删除数据_finish = _start + n;}else{if (n > capacity())reserve(n);while (_finish != _start + n){*_finish = val;++_finish;}}}void reserve(size_t n){if (n > capacity()){size_t sz = size();T* tmp = new T[n];if (_start){for (size_t i = 0; i < sz; ++i){tmp[i] = _start[i];}delete[] _start;}_start = tmp;_finish = _start + sz;_end_of_storage = _start + n;}}void push_back(const T& x){if (_finish == _end_of_storage){reserve(capacity() == 0 ? 4 : capacity() * 2);}*_finish = x;++_finish;}void pop_back(){assert(!empty());--_finish;}iterator insert(iterator pos, const T& val){assert(pos >= _start);assert(pos <= _finish);if (_finish == _end_of_storage){size_t len = pos - _start;reserve(capacity() == 0 ? 4 : capacity() * 2);// 扩容后更新pos,解决pos失效的问题pos = _start + len;}iterator end = _finish - 1;while (end >= pos){*(end + 1) = *end;--end;}*pos = val;++_finish;return pos;}iterator erase(iterator pos){assert(pos >= _start);assert(pos < _finish);iterator start = pos + 1;while (start != _finish){*(start - 1) = *start;++start;}--_finish;return pos;}size_t capacity() const{return _end_of_storage - _start;}size_t size() const{return _finish - _start;}bool empty(){return _start == _finish;}T& operator[](size_t pos){assert(pos < size());return _start[pos];}const T& operator[](size_t pos) const{assert(pos < size());return _start[pos];}private:iterator _start = nullptr;iterator _finish = nullptr;iterator _end_of_storage = nullptr;};

http://www.ppmy.cn/news/75713.html

相关文章

leetcode 884. 两句话中的不常见单词

文章目录 题目描述解题思路法1 执行结果法1 leetcode 884. 两句话中的不常见单词 题目描述 两句话中的不常见单词 句子 是一串由空格分隔的单词。每个 单词 仅由小写字母组成。 如果某个单词在其中一个句子中恰好出现一次&#xff0c;在另一个句子中却 没有出现 &#xff0c…

验证码发明人的天才故事

验证码发明人的天才故事 Luis von Ahn&#xff0c;卡内基梅隆大学计算机科学副教授&#xff0c;天才计算机科学家和企业家 01 调皮捣蛋、不循规蹈矩的青少年时代 1978年8月19日&#xff0c;Luis von Ahn出生在危地马拉首都危地马拉城的一个中产阶级家庭。 他是德国和危地马拉…

MybatisPlus--基础入门!真滴方便

目录 一、简介 2.特性 二、入门 1.创建springboot 项目(点击查看如何创建 ) 注意&#xff1a;引入 MyBatis-Plus 之后请不要再次引入 MyBatis 以及 MyBatis-Spring&#xff0c;以避免因版本差异导致的问题 2.数据准备 3.配置application.yml 4.代码 BaseMapper<>…

使用Spring Cloud构建Serverless应用

使用Spring Cloud构建Serverless应用 一、Serverless简介1. 概念介绍2. 特点及优势3. 应用场景 二、Spring Cloud简介1. 概念介绍2. Spring Cloud框架架构3. 主要功能 三、Spring Cloud构建Serverless应用1. Serverless应用与Spring Cloud的结合2. Spring Cloud Function的安装…

[深度好文]10张图带你轻松理解关系型数据库系统的工作原理

[深度好文]10张图带你轻松理解关系型数据库系统的工作原理 原文(欢迎关注)&#xff1a;https://mp.weixin.qq.com/s/CNCfWRpv8QlICGvZkLG4Jw 尽管数据库在我们应用程序中扮演着储存几乎所有状态的关键角色&#xff0c;但人们对其运行原理的了解通常仅停留在较为浅显的层面&…

渗透模拟环境配置和工具介绍-渗透测试模拟环境(0)

我们在 渗透攻防环境搭建与攻防知识体系思维导图 一文中有整体渗透环境设计和说明,但渗透环境攻击路径是什么、网络配置怎么配置,渗透环境的渗透工具是哪些、有什么作用并没有介绍,本篇将继续展开环境功能演示(环境配置步骤和验证相对较多),为模拟环境攻防系列打好基础。…

中原雄狮官网上线 | LTD物流服务行业案例分享

​一、公司介绍 中原雄狮崛起于2017年&#xff0c;彼时&#xff0c;全国货运行业存在许多不良行为&#xff0c;无赖货主和黑心货站恶意拖欠货车司机运费&#xff0c;而货车司机作为弱势群体却势单力薄无依无助的问题&#xff0c;为了让司机的血汗钱能颗粒归仓&#xff0c;中原雄…

实现图形算法API[软光栅渲染器,C++]

最近有点烦&#xff0c;发烧感冒了三天[事实上是俩天&#xff0c;第三天是因为摆得太舒服了索性多玩一天]&#xff0c;啥都没学&#xff0c;打守望先锋也把把被虐...&#xff0c;想着今天来提起键盘把之前的东西都总结一下。 那么话归真题&#xff0c;首先我是仿造opengl来写的…