【C++】— 掌握STL vector 类:“Vector简介:动态数组的高效应用”
文章目录
- 1.vector的介绍和使用
- 1.1vector的介绍
- 1.2 vector的特点
- 1.3vector的使用
- 1.3.1vector的定义
- 1.3.2vector iterator的使用
- 1.3.3vector 的空间增长问题
- 1.3.4 vector 的增删查改
- 1.3.5vector 迭代器失效问题

1.vector的介绍和使用
1.1vector的介绍
vector是一个顺序容器,可以看作是能够动态增长和缩小的数组。与普通的数组不同的是,vector在需要时可以自动调整其大小,以容纳新添加的元素。因此vector在使用上更加的灵活和方便,
1.2 vector的特点
1.动态性:vector能够根据需要动态的调整其空间的大小,这使得vector在处理不确定数量的数据时非常有用。
2.随机访问: 与数组类似,vector支持通过下标进行数据的快速访问,这意味着可以在常数时间内访问vector中的任何元素。
3.尾部操作高效: 在vector尾部添加或者删除数据时非常高效的,因为这些操作在常数时间内就能完成。
1.3vector的使用
1.3.1vector的定义
| 构造函数说明 | 接口说明 |
|---|---|
| vector() | 无参构造 |
| vector(size_type n,const value_type& val = value_type()) | 构造并初始化n个value |
| vector(const vector& x) | 拷贝构造 |
| ector(inputlterator first,inputlterator last) | 使用迭代器进行初始化构造 |
代码演示:
1.默认构造函数
std::vector<T> v;
• 创建一个空的 vector,其元素类型为T。
• 初始时,v的大小为0容量也是未定义的(通常为0,具体取决于实现)
2.指定大小的构造函数
std::vector<T> v(n);
• 创建一个大小为n的vector,其元素被默认构造(即调用T())。
• 如果T是一个类型,并没有默认构造函数,则会编译报错。
3.指定大小和初始值的构造函数
std::vector<T> v(n,val);
• 创建一个大小为n的vector,并使用val初始化每个元素。
• 这对需要初始化所有元素为相同值的场景非常有用。
4.范围构造函数
std::vector<T> v(first,last);
• 创建一个vector,并使用迭代器范围[first,last)内的元素来初始化它。
• first和last是输入迭代器,它们指向某种容器(数组、另一个vector等)中的元素。
• 注意: first和last是左闭右开的,即包含first指向的元素,但不包括last指向的元素。
5.拷贝构造函数
std::vector<T> v2(v1);
• 创建一个新的vector,它是现有vector v1的一个副本。
• 新vector 将包含与v1 相同数量和顺序的元素。
6.移动构造函数(C++11 及以后)
std::vector<T> v2(std::move(v1));
• 创建一个新的vector,并通过移动v1 的内容来初始化它。
• 这通常比拷贝构造函数更加高效,因为它可以避免不必要的复制操作。
• 然而,v1在移动后将处于未定义状态,通常不再包含有效的数据。
7.初始化列表构造函数(C++11 及以后)
std::vector<T> v1 = {val1,val2,val3,...};
或者
std::vector<T> v1{val1,val2,val3,...};
• 使用初始化列表来创建并初始化一个vector.
• 这就允许你直接在构造函数中指定要包含在vector中的元素。
1.3.2vector iterator的使用
| iterator的使用 | 接口说明 |
|---|---|
| begin + end | 获取第一个数据位置的 iterator / const_iterator;获取最后一个数据下一个位置的 iterator / const_iterator |
| rbegin + rend | 获取最后一个数据的位置的 reverse_iterator;获取第一个为数据前一个位置的 revesr_iterator |


1.3.3vector 的空间增长问题
| 接口名称 | 接口说明 |
|---|---|
| size | 获取数据的个数 |
| capacity | 获取容量大小 |
| empty | 判断是否为空 |
| resize | 改变vector的size |
| reserve | 改变vector的capacity |
1.3.4 vector 的增删查改
| vector增删查改 | 接口说明 |
|---|---|
| push_back | 尾插 |
| popop_back | 尾删 |
| find | 查找(这个是算法模块实现,不是vector的接口) |
| insert | 在pos之前插入数据val |
| erase | 删除pos位置的数据 |
| swap | 交换两个vector的数据空间 |
| operator[] | 像数组一样访问 |
代码演示:
#include<iostream>
#include<vector>
using namespace std;int main()
{//创建一个空的vectorvector<int> v;//增加元素v.push_back(10);v.push_back(20);v.insert(v.begin() + 1,15); //在第二个位置插入15// 打印 vector 中的元素cout << "Vector after insertions: ";for (int i = 0; i < v.size(); ++i) {std::cout << v[i] << " ";}cout << endl;// 查找元素auto it = find(v.begin(), v.end(), 20);if (it != v.end()) {cout << "Found 20 at position: " << distance(v.begin(), it) << endl;}else {cout << "20 not found in the vector." << endl;}// 修改元素v[1] = 18; // 将第二个元素修改为 18// 打印修改后的 vector 中的元素cout << "Vector after modification: ";for (int i = 0; i < v.size(); ++i) {cout << v[i] << " ";}cout << endl;// 删除元素(删)v.erase(find(v.begin(), v.end(), 10)); // 删除第一个值为 10 的元素v.pop_back(); // 删除末尾的元素// 打印删除后的 vector 中的元素cout << "Vector after deletions: ";for (int i = 0; i < v.size(); ++i) {cout << v[i] << " ";}cout << endl;return 0;
}
运行这段代码将输出:

1.3.5vector 迭代器失效问题
迭代器的主要作用就是让算法能够不用关系底层数据结构,其底层实际就是一个类似于指针的东西,或者对指针进行了封装。比如:vector的迭代器就是原生态指针 T*。因此迭代器失效,实际就是迭代器底层对应指针指向的空间被销毁了,而使用一块已经被释放的空间,造成的后果是程序崩溃(即如果继续使用已经失效的迭代器,程序可能会崩溃)。
对于vector可能导致其迭代器失效的操作有:
1.插入操作导致内存重新分配:
•当向vector中添加元素,且当前容量不足以容纳新的元素时,vector可能会重新分配其内存空间(即分配更大的内存块,并将现有元素复制到新的位置)。这种情况下,所有指向原vector元素的迭代器、指针和引用都会失效。
•插入操作包括push_back、emplace_back、insert等。
2.删除操作:
•删除元素(使用erase方法)会使指向被删除元素及其之后的元素的迭代器失效。这是因为删除操作会移动后续元素来填补被删除元素的位置。
3.改变vector的大小:
•使用resize方法改变vector的大小,如果新大小大于当前大小,并导致内存重新分配,那么所有迭代器都会失效。
4.清空vector:
• 使用clear方法清空vector会使所有的迭代器失效,因为所有元素都被移除了。
如何避免迭代器失效
• 预留空间: 如果知道将要插入的元素的数量,可以使用reserve方法预先分配足够的空间。这可以减少内存重新分配的可能性,从而避免迭代器失效。
• 使用返回值: 一些vector成员函数(如insert和erase)会返回指向新元素位置或者下一个有效元素的迭代器。使用这些返回值可以避免因操作导致的迭代器失效问题。
• 重新获取迭代器: 在可能导致迭代器失效的操作后,重新获取迭代器。
示例代码:
#include<iostream>
#include<vector>
using namespace std;int main()
{vector<int> v = {1,2,3,4,5};//获取初始迭代器auto it = v.begin();//插入元素,可能导致迭代器失效v.insert(v.begin(), 0);//使用重新获取的迭代器it = v.begin();//重新获取迭代器cout << "插入后的第一个元素:" << *it << endl;//删除元素,可能导致迭代器失效v.erase(it);//使用erase的返回值it = v.erase(v.begin()); //删除第一个元素,并获取新的迭代器cout << "删除后的第一个元素:" << *it << endl;return 0;
}
总之,使用 vector 时需要特别注意迭代器的有效性,尤其是在进行插入和删除操作时。通过预留空间、使用函数返回值和适时重新获取迭代器,可以有效地管理迭代器的生命周期,避免潜在的失效问题。
相关文章:
【C++】— 掌握STL vector 类:“Vector简介:动态数组的高效应用”
文章目录 1.vector的介绍和使用1.1vector的介绍1.2 vector的特点1.3vector的使用1.3.1vector的定义1.3.2vector iterator的使用1.3.3vector 的空间增长问题1.3.4 vector 的增删查改1.3.5vector 迭代器失效问题 1.vector的介绍和使用 1.1vector的介绍 vector是一个顺序容器&am…...
Docker__持续更新......
Docker 1. 基本知识1.1 为什么有Docker?1.2 Docker架构与容器化 画图解释 画图解释2. 项目实战 1. 基本知识 1.1 为什么有Docker? 用一行命令跨平台安装项目,在不同平台上运行项目。把项目打包分享运行应用。 1.2 Docker架构与容器化 准备机器,在机…...
【R语言】主成分分析与因子分析
一、主成分分析 主成分分析(Principal Component Analysis, PCA)是一种常用的无监督数据降维技术,广泛应用于统计学、数据科学和机器学习等领域。它通过正交化线性变换将(高维)原始数据投影到一个新的坐标系ÿ…...
ROS-相机话题-获取图像-颜色目标识别与定位-目标跟随-人脸检测
文章目录 相机话题获取图像颜色目标识别与定位目标跟随人脸检测 相机话题 启动仿真 roslaunch wpr_simulation wpb_stage_robocup.launch rostopic hz /kinect2/qhd/image_color_rect/camera/image_raw:原始的、未经处理的图像数据。 /camera/image_rectÿ…...
STM32 如何使用DMA和获取ADC
目录 背景 摇杆的原理 程序 端口配置 ADC 配置 DMA配置 背景 DMA是一种计算机技术,允许某些硬件子系统直接访问系统内存,而不需要中央处理器(CPU)的介入,从而减轻CPU的负担。我们可以通过DMA来从外设…...
【JAVA实战】JAVA实现Excel模板下载并填充模板下拉选项数据
背景 有这样一个场景:前端下载Excel模板,进行数据导入,这个下载模板过程需要经过后端接口去数据库查询数据进行某些列的下拉数据填充,下拉填充的数据过程中会出现错误String literals in formulas can’t be bigger than 255 cha…...
java面试笔记(一)
1. 一万个string类型的数据,设计一个算法如何按照String长度来排序 以使用 Arrays.sort() 方法,并结合一个自定义的比较器。以下是实现的示例代码: public class StringLengthSort {public static void main(String[] args) {// 定义一万个字符串的示例…...
【C++】36.C++IO流
文章目录 1. C语言的输入与输出2. 流是什么3. CIO流3.1 C标准IO流3.2 C文件IO流 4. stringstream的简单介绍 1. C语言的输入与输出 C语言中我们用到的最频繁的输入输出方式就是scanf ()与printf()。 scanf(): 从标准输入设备(键盘)读取数据,并将值存放在变量中。pri…...
Qt5开发入门指南:从零开始掌握跨平台开发
目录 Qt框架概述 开发环境搭建 基础语法与核心机制 第一个Qt窗口程序 常见问题解答 一、Qt框架概述 1.1 什么是Qt? Qt是一个1995年由挪威Trolltech公司开发的跨平台C图形用户界面应用程序框架。最新Qt5版本主要包含: GUI模块:支持Wind…...
Rook-ceph(1.92最新版)
安装前准备 #确认安装lvm2 yum install lvm2 -y #启用rbd模块 modprobe rbd cat > /etc/rc.sysinit << EOF #!/bin/bash for file in /etc/sysconfig/modules/*.modules do[ -x \$file ] && \$file done EOF cat > /etc/sysconfig/modules/rbd.modules &l…...
深度学习在蛋白质-蛋白质相互作用(PPI)领域的研究进展(2022-2025)
一、蛋白质-蛋白质相互作用(PPI)的定义与生物学意义 蛋白质-蛋白质相互作用(Protein-Protein Interaction, PPI)是指两个或多个蛋白质通过物理结合形成复合物,进而调控细胞信号传导、代谢、免疫应答等生命活动的过程。PPI是生物体内复杂功能网络的核心,例如酶与底物的结…...
网络安全学习架构 网络安全架构内容
网上买的《信息安全原理及应用》的书还没到,就找了一本《密码编码学与网络安全》的电子书,写的也不错,计划今天和明天把第一章绪论和第二章的数论给看完 1. 计算机网络安全概念 计算机安全的三个核心是:完整性(只要特…...
硕成C语言24
不管风吹浪打,胜似闲庭信步! 1.选择排序: 不管风吹浪打,胜似闲庭信步!1.选择排序: // 这个宏定义用于忽略 Visual Studio 中一些安全相关的警告,使得可以使用像 scanf 这样的函数而不产生警告…...
《Stable Diffusion绘画完全指南:从入门到精通的Prompt设计艺术》-配套代码示例
第一章:模型加载与基础生成 1.1 基础模型加载 from diffusers import StableDiffusionPipeline import torch# 加载SD 1.5基础模型(FP32精度) pipe StableDiffusionPipeline.from_pretrained("runwayml/stable-diffusion-v1-5",…...
Linux下为Intel核显安装OpenCL
实验环境 CPU:N5105运行内存:32G系统:almalinux 8 (类似CentOS系统) 安装过程 检查i915驱动 输入命令dmesg | grep i915查看驱动的加载信息。 我这里没接显示器,输出正确的驱动加载信息如下图所示&…...
用deepseek学大模型04-机器学习建模过程
https://metaso.cn/s/7tEmBHr 目前已经学完深度学习的数学基础,给出机器学习建模的基本过程 好的,我现在需要回答用户的问题:已经学完深度学习的数学基础,现在要给出机器学习建模的基本过程。我需要结合我搜索到的资料来构建一个详…...
【ClickHouse】Ubuntu下离线安装ClickHouse数据库并使用DBeaver连接
目录 0. 安装前准备1 安装ClickHouse1.1 下载安装包1.2 离线安装1.3 配置密码1.4 启动ClickHouse服务 2 DBeaver连接配置2.1 下载ClickHouse驱动2.2 DBeaver配置2.2.1 配置主要参数2.2.2 配置驱动 2.3 常见问题处理2.3.1 修改远程登录配置2.3.2 更新驱动配置 0. 安装前准备 有…...
Unity3D实现接入DeepSeek对话
系列文章目录 unity知识点 文章目录 系列文章目录👉前言👉一、使用第一步👉1-1、登录注册DeepSeek👉1-2、创建API-key👉二、使用第二步👉三、使用第三步👉壁纸分享👉总结👉前言 随着人工智能和机器学习技术的不断进步,DeepSeek的未来发展趋势充满了无限可能…...
【ISO 14229-1:2023 UDS诊断(会话控制0x10服务)测试用例CAPL代码全解析②】
ISO 14229-1:2023 UDS诊断【会话控制0x10服务】_TestCase02 作者:车端域控测试工程师 更新日期:2025年02月15日 关键词:UDS诊断、0x10服务、诊断会话控制、ECU测试、ISO 14229-1:2023 TC10-002测试用例 用例ID测试场景验证要点参考条款预期…...
前端新手必看:10 大 UI 组件库全面解析,快速搭建高质量 Web 应用」 「从零开始:Vue 和 React 最受欢迎的 UI 组件库入门指南」 「超实用!PC 端和移动端 UI 组件库推荐与实战
前端新手必看:10 大 UI 组件库全面解析,快速搭建高质量 Web 应用 目录 什么是 UI 组件库?为什么需要 UI 组件库?PC 端 UI 组件库推荐 Ant DesignElement UIVuetifyBootstrapVueiView (View UI)Quasar FrameworkMaterial-UI (MUI…...
基于算法竞赛的c++编程(28)结构体的进阶应用
结构体的嵌套与复杂数据组织 在C中,结构体可以嵌套使用,形成更复杂的数据结构。例如,可以通过嵌套结构体描述多层级数据关系: struct Address {string city;string street;int zipCode; };struct Employee {string name;int id;…...
零门槛NAS搭建:WinNAS如何让普通电脑秒变私有云?
一、核心优势:专为Windows用户设计的极简NAS WinNAS由深圳耘想存储科技开发,是一款收费低廉但功能全面的Windows NAS工具,主打“无学习成本部署” 。与其他NAS软件相比,其优势在于: 无需硬件改造:将任意W…...
Cursor实现用excel数据填充word模版的方法
cursor主页:https://www.cursor.com/ 任务目标:把excel格式的数据里的单元格,按照某一个固定模版填充到word中 文章目录 注意事项逐步生成程序1. 确定格式2. 调试程序 注意事项 直接给一个excel文件和最终呈现的word文件的示例,…...
【入坑系列】TiDB 强制索引在不同库下不生效问题
文章目录 背景SQL 优化情况线上SQL运行情况分析怀疑1:执行计划绑定问题?尝试:SHOW WARNINGS 查看警告探索 TiDB 的 USE_INDEX 写法Hint 不生效问题排查解决参考背景 项目中使用 TiDB 数据库,并对 SQL 进行优化了,添加了强制索引。 UAT 环境已经生效,但 PROD 环境强制索…...
线程同步:确保多线程程序的安全与高效!
全文目录: 开篇语前序前言第一部分:线程同步的概念与问题1.1 线程同步的概念1.2 线程同步的问题1.3 线程同步的解决方案 第二部分:synchronized关键字的使用2.1 使用 synchronized修饰方法2.2 使用 synchronized修饰代码块 第三部分ÿ…...
STM32标准库-DMA直接存储器存取
文章目录 一、DMA1.1简介1.2存储器映像1.3DMA框图1.4DMA基本结构1.5DMA请求1.6数据宽度与对齐1.7数据转运DMA1.8ADC扫描模式DMA 二、数据转运DMA2.1接线图2.2代码2.3相关API 一、DMA 1.1简介 DMA(Direct Memory Access)直接存储器存取 DMA可以提供外设…...
Maven 概述、安装、配置、仓库、私服详解
目录 1、Maven 概述 1.1 Maven 的定义 1.2 Maven 解决的问题 1.3 Maven 的核心特性与优势 2、Maven 安装 2.1 下载 Maven 2.2 安装配置 Maven 2.3 测试安装 2.4 修改 Maven 本地仓库的默认路径 3、Maven 配置 3.1 配置本地仓库 3.2 配置 JDK 3.3 IDEA 配置本地 Ma…...
Cilium动手实验室: 精通之旅---13.Cilium LoadBalancer IPAM and L2 Service Announcement
Cilium动手实验室: 精通之旅---13.Cilium LoadBalancer IPAM and L2 Service Announcement 1. LAB环境2. L2公告策略2.1 部署Death Star2.2 访问服务2.3 部署L2公告策略2.4 服务宣告 3. 可视化 ARP 流量3.1 部署新服务3.2 准备可视化3.3 再次请求 4. 自动IPAM4.1 IPAM Pool4.2 …...
智能职业发展系统:AI驱动的职业规划平台技术解析
智能职业发展系统:AI驱动的职业规划平台技术解析 引言:数字时代的职业革命 在当今瞬息万变的就业市场中,传统的职业规划方法已无法满足个人和企业的需求。据统计,全球每年有超过2亿人面临职业转型困境,而企业也因此遭…...
前端调试HTTP状态码
1xx(信息类状态码) 这类状态码表示临时响应,需要客户端继续处理请求。 100 Continue 服务器已收到请求的初始部分,客户端应继续发送剩余部分。 2xx(成功类状态码) 表示请求已成功被服务器接收、理解并处…...
