C++ STL vector容器详解:从原理到实践
引言
亲爱的小伙伴们,今天我要和大家分享一个C++编程中的"神器"——vector容器!作为STL(标准模板库)中最常用的容器之一,vector就像是一个"超级数组",既有数组的高效随机访问特性,又能自动调整大小,让你告别固定大小数组的困扰。不管你是刚接触C++的新手,还是想深入理解vector实现原理的老手,这篇博客都能带给你新的收获!
一、什么是vector?
想象一下,你正在收集各种小玩具,但不确定最终会有多少个。用普通数组的话,你得提前决定一个上限,太小了不够用,太大了又浪费空间。这时候,vector就像一个神奇的收纳盒,它能根据你放入的玩具数量自动调整大小!
定义与特点
- 动态数组:vector是一个能够存放任意类型的动态数组
- 连续内存:vector中的元素在内存中连续存储,支持高效的随机访问
- 自动扩容:当空间不足时,vector会自动分配更大的内存空间
- 类型安全:通过模板机制确保类型安全
vector的内部结构
从实现原理上看,vector内部维护三个关键指针:
指向数据的起始位置 —— start
指向最后一个元素的下一个位置 —— finish
指向分配的内存空间的尾端 —— end_of_storage
!vector内部结构示意图
这就像一个能伸缩的宿舍:
- start是宿舍的门口
- finish是最后一名同学睡的床位后面一点的位置
- end_of_storage是宿舍的墙壁(容量上限)
二、vector的基本操作
创建vector
#include <vector>// 创建空vectorstd::vector<int> vec1;// 创建包含5个元素的vector,所有元素初始化为0std::vector<int> vec2(5);// 创建包含5个元素的vector,所有元素初始化为10std::vector<int> vec3(5, 10);// 使用初始化列表(C++11)std::vector<int> vec4 = {1, 2, 3, 4, 5};// 从数组创建vectorint arr[] = {1, 2, 3, 4, 5};std::vector<int> vec5(arr, arr + 5);// 从另一个vector创建std::vector<int> vec6(vec5);
这就像不同的方式开party:可以先空着等人来(vec1),可以先邀请5个人(vec2),可以先邀请5个特定的朋友(vec3)...
添加和删除元素
std::vector<int> vec;// 在尾部添加元素vec.push_back(10); // 添加元素10到尾部// 在指定位置插入元素vec.insert(vec.begin() + 1, 20); // 在第二个位置插入20// 删除尾部元素vec.pop_back();// 删除指定位置的元素vec.erase(vec.begin() + 1); // 删除第二个元素// 清空所有元素vec.clear();
想象一下排队买票:
- push_back是有人来排队,自动站到队尾
- insert是有人插队到指定位置
- pop_back是最后一个人不想排了,离开了
- erase是队伍中间某个人离开了
- clear是突然下大雨,所有人都散了
访问元素
std::vector<int> vec = {10, 20, 30, 40, 50};// 使用索引访问int a = vec[2]; // 获取索引为2的元素(30)// 使用at函数(带边界检查)int b = vec.at(3); // 获取索引为3的元素(40),越界会抛出异常// 获取第一个和最后一个元素int first = vec.front(); // 获取第一个元素(10)int last = vec.back(); // 获取最后一个元素(50)
这就像找教室里的同学:直接叫号(索引)、点名确认(at)、找第一个和最后一个。
三、vector的容量管理
理解vector的容量管理是掌握其高效使用的关键。
size与capacity
- size:当前vector中实际元素的个数
- capacity:当前vector在不扩容的情况下最多能容纳的元素个数
std::vector<int> vec;for(int i = 0; i < 10; i++) {vec.push_back(i);std::cout << "Size: " << vec.size() << ", Capacity: " << vec.capacity() << std::endl;}
可能的输出结果:
Size: 1, Capacity: 1Size: 2, Capacity: 2Size: 3, Capacity: 4Size: 4, Capacity: 4Size: 5, Capacity: 8Size: 6, Capacity: 8Size: 7, Capacity: 8Size: 8, Capacity: 8Size: 9, Capacity: 16Size: 10, Capacity: 16
这就像一个聚会场地:
- size是实际到场的人数
- capacity是场地能容纳的最大人数
- 当人数超过容量时,需要换一个更大的场地
扩容原理
当vector需要更多空间时:
- 分配一块更大的内存(通常是当前容量的1.5倍或2倍)
- 将现有元素复制到新内存
- 释放旧内存
- 更新内部指针
这个过程叫做重新分配(reallocation),它是耗时的操作,因为需要复制所有元素。
优化容量管理
// 预留空间,避免频繁扩容vec.reserve(1000); // 预分配至少能容纳1000个元素的空间// 调整大小(会改变size)vec.resize(100); // 调整为100个元素,如果扩大则用默认值填充// 减少容量以适应当前大小vec.shrink_to_fit(); // C++11引入
- reserve是提前租一个足够大的场地,避免中途换场地
- resize是主动调整聚会人数
- shrink_to_fit是在人到齐后,换一个刚好够用的小场地,节省费用
四、vector的迭代器
迭代器是访问容器元素的统一接口,就像"智能指针"。
std::vector<int> vec = {10, 20, 30, 40, 50};// 使用迭代器遍历for(std::vector<int>::iterator it = vec.begin(); it != vec.end(); ++it) {std::cout << *it << " "; // 输出:10 20 30 40 50}// C++11引入的范围for循环(语法糖)for(auto& element : vec) {std::cout << element << " "; // 输出同上}// 反向迭代器for(std::vector<int>::reverse_iterator rit = vec.rbegin(); rit != vec.rend(); ++rit) {std::cout << *rit << " "; // 输出:50 40 30 20 10}
这就像参观博物馆:
- 普通迭代器是从入口到出口的参观路线
- 反向迭代器是从出口回到入口的路线
- 范围for循环就是跟着导游走,不用担心路线
五、vector的常见陷阱与优化
1. 迭代器失效问题
当vector扩容或删除元素时,迭代器可能会失效:
std::vector<int> vec = {1, 2, 3, 4, 5};auto it = vec.begin() + 2; // 指向元素3vec.insert(vec.begin(), 0); // 可能导致扩容// 此时it已失效,使用it将导致未定义行为
解决方法:在可能导致迭代器失效的操作后,重新获取迭代器。
2. 不必要的复制
// 低效的循环添加元素std::vector<int> vec;for(int i = 0; i < 10000; i++) {vec.push_back(i); // 可能多次扩容}// 优化:预先分配足够空间std::vector<int> vec2;vec2.reserve(10000); // 提前分配空间for(int i = 0; i < 10000; i++) {vec2.push_back(i); // 避免扩容}
3. 返回值优化
// 避免不必要的复制void processVector(const std::vector<int>& vec) {// 使用const引用作为参数}std::vector<int> createVector() {std::vector<int> result = {1, 2, 3};return result; // 编译器会优化,避免不必要的复制}
六、实际应用案例
案例1:动态二维数组
// 创建10x20的二维数组std::vector<std::vector<int>> matrix(10, std::vector<int>(20, 0));// 访问元素matrix[5][10] = 100;// 动态调整行数matrix.resize(15); // 现在是15x20
案例2:高效字符串分割
std::vector<std::string> split(const std::string& str, char delimiter) {std::vector<std::string> tokens;std::string token;std::istringstream tokenStream(str);while (std::getline(tokenStream, token, delimiter)) {tokens.push_back(token);}return tokens;}
案例3:自定义对象的存储
struct Student {std::string name;int age;Student(const std::string& n, int a) : name(n), age(a) {}};std::vector<Student> students;students.push_back(Student("小明", 18));students.push_back(Student("小红", 17));// C++11的emplace_back可以在容器内直接构造对象,避免复制students.emplace_back("小刚", 19);
总结
vector是C++中最常用的容器之一,它提供了数组的高效随机访问,又克服了数组大小固定的限制。通过深入理解vector的工作原理,你可以更高效地使用它,避免常见陷阱,写出更优雅高效的代码。
记住:合理使用reserve预分配空间、避免不必要的复制、注意迭代器失效问题,这些都是使用vector的关键技巧!
在下一篇博客中,我们将更深入地探讨vector的内存管理机制和更多高级用法。敬请期待!
相关文章:
C++ STL vector容器详解:从原理到实践
引言 亲爱的小伙伴们,今天我要和大家分享一个C编程中的"神器"——vector容器!作为STL(标准模板库)中最常用的容器之一,vector就像是一个"超级数组",既有数组的高效随机访问特性&#…...
视频压制(Video Encoding/Compression)
视频压制(Video Encoding/Compression) 视频压制是指通过特定的算法和技术,将原始视频文件转换为更小体积或更适合传播的格式的过程。其核心目的是在尽量保持画质的前提下,减少视频的文件大小,或适配不同播放设备、网络环境的需求…...

【论文笔记】Transcoders Find Interpretable LLM Feature Circuits
Abstract 机制可解释性(mechanistic interpretability)的核心目标是路径分析(circuit analysis):在模型中找出与特定行为或能力对应的稀疏子图。 然而,MLP 子层使得在基于 Transformer 的语言模型中进行细粒度的路径分析变得困难。具体而言,…...
音视频融合中的语音分离技术实现
音视频融合中的语音分离技术实现 一、任务概述 语音分离是音频信号处理的核心任务,旨在从混合音频中分离出目标语音。音视频融合技术通过结合视觉信息(如嘴唇运动)显著提升分离效果。本方案将实现一个基于深度学习的音视频融合语音分离系统。 二、系统架构 #mermaid-svg-3…...

每天总结一个html标签——a标签
文章目录 一、定义与使用说明二、支持的属性三、支持的事件四、默认样式五、常见用法1. 文本链接2. 图片链接3. 导航栏 在前端开发中,a标签(锚点标签)是最常用的HTML标签之一,主要用于创建超链接,实现页面间的跳转或下…...
在Babylon.js中创建3D文字:简单而强大的方法
引言 在3D场景中添加文字是许多WebGL项目的常见需求。Babylon.js提供了多种创建3D文字的方法,其中使用TextBlock结合平面网格是一种简单而高效的方式。本文将介绍如何使用Babylon.js的GUI系统在3D空间中创建美观的文字效果。 方法概述 Babylon.js的GUI系统允许我…...
CSS 渐变完全指南:从基础概念到实战案例(线性渐变/径向渐变/重复渐变)
一、什么是 CSS 渐变? 渐变是网页设计中常用的视觉效果,指两种或多种颜色之间的平滑过渡。CSS 提供了强大的渐变功能,无需依赖图片即可创建复杂的色彩过渡效果,主要分为线性渐变和径向渐变两大类。 二、线性渐变(Line…...
初识Docker:容器化技术的入门指南
初识Docker:容器化技术的入门指南 一、Docker是什么:容器化技术的核心概念二、Docker的核心优势2.1 环境一致性2.2 高效部署与快速迭代2.3 资源利用率高 三、Docker的安装与基本使用3.1 安装Docker3.2 Docker基本概念3.3 第一个Docker容器体验 四、Docke…...

android binder(1)基本原理
一、IPC 进程间通信(IPC,Inter-Process Communication)机制,用于解决不同进程间的数据交互问题。 不同进程之间用户地址空间的变量和函数是不能相互访问的,但是不同进程的内核地址空间是相同和共享的,我们可…...

行业分析---小米汽车2025第一季度财报
1 背景 最近几年是新能源汽车的淘汰赛,前短时间比亚迪再次开始了降价,导致一片上市车企的股价大跌,足见车圈现在的敏感度。因此笔者会一直跟踪新势力车企的财报状况,对之前财报分析感兴趣的读者朋友可以参考以下博客:…...

边缘计算网关支撑医院供暖系统高效运维的本地化计算与边缘决策
一、项目背景 医院作为人员密集的特殊场所,对供暖系统的稳定性和高效性有着极高的要求。其供暖换热站传统的人工现场监控方式存在诸多弊端,如人员值守成本高、数据记录不及时不准确、故障发现和处理滞后、能耗难以有效监测和控制等,难以满足…...
GO环境配置
Go 语言环境安装指南(Windows 版) 以下是在 Windows 系统上安装 Go 语言环境的完整步骤: 准备工作 操作系统要求:Windows 7 或更高版本(推荐 Windows 10/11)系统架构:64位(…...
`docker run`、`docker start`、`docker exec` 区别
🧠 先给你一句话理解: docker run ≈ docker create docker start docker exec(第一次) ✅ 三者的区别一览表 命令作用类比真实生活常用场景docker run创建 启动 执行命令(一次性)你买了一台新电脑&am…...

简单了解string类的特性及使用(C++)
string的特性 string类不属于STL,它属于标准库 但由于它具有数据结构的特性,所以从归类的角度,可以将string类归类到容器里面去 在C标准库中,std::string 是一个特化的类型,实际上是 std::basic_string 的别名。std…...

FastAPI+Pyomo实现线性回归解决饮食问题
之前在 FastAPI介绍-CSDN博客 中介绍过FastAPI,在 Pyomo中线性规划接口的使用-CSDN博客 中使用Pyomo解决饮食问题,这里将两者组合,即FastAPI在服务器端启动,通过Pyomo实现线性回归;客户端通过浏览器获取饮食的最优解。…...

16.FreeRTOS
目录 第1章 FreeRTOS 实时操作系统 1.1 认识实时操作系统 1.1.1 裸机的概念 1.1.2 操作系统的概念 1.2 操作系统的分类 1.3 常见的操作系统 1.4 认识实时操作系统 1.4.1 可剥夺型内核与不可剥夺型内核 1.4.2 嵌入式操作系统的作用 1.4.3 嵌入式操作系统的发展 1.4.4…...

Redis最佳实践——购物车优化详解
Redis在电商购物车高并发读写场景下的优化实践 一、购物车业务场景分析 典型操作特征 读/写比例 ≈ 8:2高峰QPS可达10万单用户最大商品数500操作类型:增删改查、全选/反选、数量修改 技术挑战 高并发下的数据一致性海量数据存储与快速访问实时价格计算与库存校验分…...

【计算机网络】传输层UDP协议
🔥个人主页🔥:孤寂大仙V 🌈收录专栏🌈:计算机网络 🌹往期回顾🌹: 【计算机网络】应用层协议Http——构建Http服务服务器 🔖流水不争,争的是滔滔不…...

安全漏洞修复导致SpringBoot2.7与Springfox不兼容
项目基于 springboot2.5.2 实现的,用 springfox-swagger2 生成与前端对接的 API 文档;pom.xml 中依赖如下 <parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId>&l…...

从法律层面剖析危化品证书:两证一证背后的安全逻辑
《安全生产法》第 24 条明确规定,危化品单位主要负责人和安全管理人员 “必须考核合格方可上岗”。这并非仅仅是行政要求,而是通过法律来筑牢安全防线。在某危化品仓库爆炸事故中,由于负责人未持证,导致事故责任升级,企…...
C语言——获取变量所在地址(uint8和uint32的区别)
前言: 1.使用uint8 *的原因 在C语言中,获取或操作一个4字节地址(指针)时使用uint8_t*(即unsigned char*)而不是uint32_t*,主要基于以下关键原因: 1.1. 避免违反严格别名规则&…...
2 Studying《Effective STL》
目录 0 引言 1 容器 1. 慎重选择容器类型 3. 确保容器中的对象副本正确且高效 4. 调用empty()而不是检查size()是否为0 5. 区间成员函数优先于与之对应的单元素成员函数 7. 如果容器中包含了通过new创建的指针,切记析构前将指针delete掉 9. 慎重选择删除元素…...

深入理解复数加法与乘法:MATLAB演示
在学习复数的过程中,复数加法与乘法是两个非常基础且重要的概念。复数的加法和乘法操作与我们常见的实数运算有所不同,它们不仅涉及到数值的大小,还有方向和相位的变化。在这篇博客中,我们将通过MATLAB演示来帮助大家更好地理解复…...

【设计模式-3.6】结构型——桥接模式
说明:本文介绍结构型设计模式之一的桥接模式 定义 桥接模式(Bridge Pattern)又叫作桥梁模式、接口(Interface)模式或柄体(Handle and Body)模式,指将抽象部分与具体实现部分分离&a…...
【前端】性能优化篇
长期更新,建议关注收藏点赞。 目录 性能优化具体指标监控工具/系统解决方案htmlcssjsvuereact包体积静态资源图片优化白屏首屏加载速度缓存优化网络优化web worker动画 面试题汇总怎么实现无限加载,如果有一亿条数据怎么优化 性能优化 本文仅是列出常见…...
【redis实战篇】第六天
摘要: 本文介绍了基于Redis的秒杀系统优化方案,主要包含两部分:1)通过Lua脚本校验用户秒杀资格,结合Java异步处理订单提升性能;2)使用Redis Stream实现消息队列处理订单。方案采用Lua脚本保证库…...

力扣题解654:最大二叉树
一、题目内容 题目要求根据一个不重复的整数数组 nums 构建最大二叉树。最大二叉树的构建规则如下: 创建一个根节点,其值为 nums 中的最大值。递归地在最大值左边的子数组前缀上构建左子树。递归地在最大值右边的子数组后缀上构建右子树。返回由 nums 构…...
手写ArrayList和LinkedList
项目仓库:https://gitee.com/bossDuy/hand-tear-collection-series 基于b站up生生大佬:https://www.bilibili.com/video/BV1Kp5tzGEc5/?spm_id_from333.788.videopod.sections&vd_source4cda4baec795c32b16ddd661bb9ce865 LinkedList package com…...
Android bindservice绑定服务,bindServiceAsUser补充
Android bindservice绑定服务,并同步返回service对象的两个方法-CSDN博客 补充反射并调用bindServiceAsUser的方法: private boolean initService2(final Context context){if(deviceServicenull){latch new CountDownLatch(1);HandlerThread handler…...
[蓝桥杯]交换次数
交换次数 题目描述 IT 产业人才需求节节攀升。业内巨头百度、阿里巴巴、腾讯(简称 BAT )在某海滩进行招聘活动。 招聘部门一字排开。由于是自由抢占席位,三大公司的席位随机交错在一起,形如:BABTATT,这使…...