当前位置: 首页 > article >正文

C++ -- vector

vector

  • 1. 关于vector
    • 1.1 对比原生数组
    • 1.2 vector的核心优势
  • 2. 扩容
    • 2.1 底层实现
    • 2.2 扩容过程
  • 3. 构造函数
  • 4. 接口模拟实现
    • 4.1 实现迭代器
    • 4.2 扩容
    • 4.3 重载[]
    • 4.4 插入和删除
    • 4.5 构造函数和析构函数
  • 5. 迭代器失效
    • 5.1 扩容后失效
    • 5.2 越界失效
  • 6. 深浅拷贝

1. 关于vector

1.1 对比原生数组

  1. 固定大小,缺乏动态扩展能力
    • 原生数组需在编译时确定大小
    • vector可根据需求动态扩容
  2. 安全性问题
    • 原生数组越界访问会导致未定义行为,甚至程序崩溃
    • vector提供边界检查
  3. 功能缺失
    • 原生数组不支持直接插入、删除或动态调整
    • vector提供丰富接口

1.2 vector的核心优势

  1. 动态数组特性:自动管理内存,无需手动new/delete;支持动态扩容
  2. 连续内存布局:数据在内存中连续存储,可通过下标直接定位元素
  3. 高性能操作:尾部插入/删除均摊O(1)时间复杂度

2. 扩容

2.1 底层实现

在这里插入图片描述
在vector上有三个迭代器:_start指向头部,_finish指向有效数据的下一个位置,_endofstorage指向存储容量的尾

2.2 扩容过程

在插入数据时,是否扩容的前提是_finish是否等于_endofstorage,如果是则会进行扩容。在这里插入图片描述
扩容时,编译器会开辟一块新空间,将原有数据拷贝到新空间上,并将迭代器指向新的空间,最后释放掉旧空间。
在这里插入图片描述

3. 构造函数

vector的构造函数有这几种
在这里插入图片描述

int main()
{vector<int> v1;vector<int> v2(10, 0);// 创建n个,并给其初始值string str = "hello world";vector<int> v3(str.begin(), str.end());// 用迭代器vector<int> v4(v3);// 拷贝构造return 0;
}

4. 接口模拟实现

template<class T>
class vector
{
public:// ......private:iterator _start; // 指向数据块的开始iterator _finish; // 指向有效数据的尾iterator _endOfStorage; // 指向存储容量的尾
};

4.1 实现迭代器

typedef T* iterator;
typedef const T* const_iterator;iterator begin()
{return _start;
}iterator end()
{return _finish;
}const_iterator cbegin() const
{return _start;
}const_iterator cend() const
{return _finish;
}

4.2 扩容

void reserve(size_t n)
{// 开辟新空间 把旧空间上的数据拷贝到新空间上 释放旧空间size_t oldsize = size();T* tmp = new T[n];// 如果原空间上有数据就要先拷贝到新空间上if (_start){memcpy(tmp, _start, sizeof(T) * size());delete[] _start;}_start = tmp;_endOfStorage = _start + n;//_finish = _start + size();// 有问题:此时的_start是新空间上的,不是原空间_finish = _start + oldsize;
}void resize(size_t n, const T& value = T())
{// 判断是否要扩容if (_finish == _endOfStorage){size_t newcapacity = capacity() == 0 ? 4 : capacity() * 2;reserve(newcapacity);}
}

4.3 重载[]

T& operator[](size_t pos)
{assert(pos < size());return _start[pos];
}const T& operator[](size_t pos)const
{return _start[pos];
}

4.4 插入和删除

void push_back(const T& x)
{if (_finish == _endOfStorage){size_t newcapacity = capacity() == 0 ? 4 : capacity() * 2;reserve(newcapacity);}*(_finish) = x;++_finish;
}void pop_back()
{--_finish;
}iterator insert(iterator pos, const T& x)
{assert(pos <= _finish);assert(pos >= _start);if (_finish == _endOfStorage){size_t len = pos - _start;size_t newcapacity = capacity() == 0 ? 4 : capacity() * 2;reserve(newcapacity);pos = _start + len;}iterator tmp = _finish - 1;while (pos <= tmp){*(tmp + 1) = *tmp;--tmp;}*pos = x;++_finish;return pos;
}iterator erase(iterator pos)
{assert(pos <= _finish);assert(pos >= _start);iterator tmp = _finish - 1;while (pos < tmp){*pos = *(pos + 1);pos++;}--_finish;return pos;
}

4.5 构造函数和析构函数

vector()
{if (_finish == _endOfStorage){size_t newcapacity = capacity() == 0 ? 4 : capacity() * 2;reserve(newcapacity);}
}vector(int n, const T& value = T())
{reserve(n);for (size_t i = 0; i < n; i++){push_back(val);}
}template<class InputIterator>
vector(InputIterator first, InputIterator last)
{while (first != last){push_back(*first);++first;}
}vector(const vector<T>& v)
{reserve(v.capacity());for (auto e : v){push_back(e);}
}vector<T>& operator= (vector<T> v)
{_start = v._start;_finish = v._finish;_endOfStorage = v._endOfStorage;
}~vector()
{if (_start){delete[] _start;_start = _finish = _endOfStorage = nullptr;}
}

5. 迭代器失效

迭代器失效常发生在扩容和越界时。

5.1 扩容后失效

void reserve(size_t n)
{// 开辟新空间 把旧空间上的数据拷贝到新空间上 释放旧空间size_t oldsize = size();T* tmp = new T[n];// 如果原空间上有数据就要先拷贝到新空间上if (_start){memcpy(tmp, _start, sizeof(T) * size());delete[] _start;}_start = tmp;_endOfStorage = _start + n;//_finish = _start + size();// 有问题:此时的_start是新空间上的,不是原空间_finish = _start + oldsize;
}

在实现插入接口时,如果不考虑记录oldsize就会出现迭代器失效问题。
在这里插入图片描述
在插入数据时,如果涉及扩容,也可能出现这种情况

iterator insert(iterator pos, const T& x)
{assert(pos <= _finish);assert(pos >= _start);if (_finish == _endOfStorage){size_t len = pos - _start;size_t newcapacity = capacity() == 0 ? 4 : capacity() * 2;reserve(newcapacity);pos = _start + len;}iterator tmp = _finish - 1;while (pos <= tmp){*(tmp + 1) = *tmp;--tmp;}*pos = x;++_finish;return pos;
}

在这里插入图片描述

5.2 越界失效

可能会发生在删除数据时
在这里插入图片描述

6. 深浅拷贝

if (_start)
{memcpy(tmp, _start, sizeof(T) * size());delete[] _start;
}

这段代码不用strcpy的原因是浅拷贝会引发二次析构。浅拷贝后的对象指向的空间和拷贝目标为同一块空间,第一次delete后这段空间消失,此时tmp相当于被析构了。但后续中,tmp还会再析构一次,造成二次析构导致程序崩溃。所以这里需要实现深拷贝。

相关文章:

C++ -- vector

vector 1. 关于vector1.1 对比原生数组1.2 vector的核心优势 2. 扩容2.1 底层实现2.2 扩容过程 3. 构造函数4. 接口模拟实现4.1 实现迭代器4.2 扩容4.3 重载[]4.4 插入和删除4.5 构造函数和析构函数 5. 迭代器失效5.1 扩容后失效5.2 越界失效 6. 深浅拷贝 1. 关于vector 1.1 对…...

GitLab-CI简介

概述 持续集成&#xff08;CI&#xff09;和 持续交付(CD) 是一种流行的软件开发实践&#xff0c;每次提交都通过自动化的构建&#xff08;测试、编译、发布&#xff09;来验证&#xff0c;从而尽早的发现错误。 持续集成实现了DevOps, 使开发人员和运维人员从繁琐的工作中解…...

深入研究Azure 容器网络接口 (CNI) overlay

启用cni overlay 在通过portal创建aks的时候,在networking配置上,选中下面的选项即可启用。 通过CLI创建AKS 要创建具有 CNI 覆盖网络的 AKS 群集,需要在创建群集时指定 --network-plugin azure 和 --network-plugin-mode 覆盖选项。 还需要指定 --pod-cidr 选项来定义群…...

Python打卡第37天

浙大疏锦行 早停策略和模型权重的保存 作业&#xff1a;对信贷数据集训练后保存权重&#xff0c;加载权重后继续训练50轮&#xff0c;并采取早停策略 import torch import torch.nn as nn import pandas as pd import matplotlib.pyplot as plt import torch.optim as optim fr…...

使用 OpenCV 构建稳定的多面镜片墙效果(镜面反射 + Delaunay 分块)

✨ 效果概览 我们将实现一种视觉效果&#xff0c;模拟由许多小镜面拼接而成的“镜子墙”。每个镜面是一个三角形区域&#xff0c;其内容做镜像反射&#xff08;如水平翻转&#xff09;&#xff0c;在视频中形成奇特的万花筒、哈哈镜、空间折叠感。 使用 OpenCV 实现“随机镜面…...

HTTP协议版本的发展(HTTP/0.9、1.0、1.1、2、3)

目录 HTTP协议层次图 HTTP/0.9 例子 HTTP/1.0 Content-Type 字段 Content-Encoding 字段 例子 1.0版本存在的问题&#xff1a;短链接、队头阻塞 HTTP/1.1 Host字段 Content-Length 字段 分块传输编码 1.1版本存在的问题 HTTP/2 HTTP/2数据传输 2版本存在的问题…...

零基础设计模式——结构型模式 - 桥接模式

第三部分&#xff1a;结构型模式 - 桥接模式 (Bridge Pattern) 在学习了适配器模式如何解决接口不兼容问题后&#xff0c;我们来看看桥接模式。桥接模式是一种更侧重于系统设计的模式&#xff0c;它旨在将抽象部分与其实现部分分离&#xff0c;使它们可以独立地变化。 核心思…...

C++对象的内存模型

C++对象的内存模型涉及对象的数据成员(包括静态成员和非静态成员)、成员函数以及虚函数表等在内存中的布局和管理方式。以下是C++对象的内存模型的主要组成部分: 1. C++对象的组成 一个C++对象通常由以下几个部分组成: 非静态数据成员 对象的核心组成部分,每个对象都有自己…...

SpringBoot3集成Oauth2.1——4集成Swagger/OpenAPI3

文章目录 访问在线文档页面配置OpenApiConfig 在我之前的文章中&#xff0c;写了 SpringBoot3集成OpenAPI3(解决Boot2升级Boot3) 访问在线文档页面 当我们同样在SpringBoot3使用oauth2.1也就是我之前的文章中写的。现在我们要处理下面这两个的问题了。 <!-- 使用springdoc…...

基于深度学习的情绪识别检测系统【完整版】

最近很多小伙伴都在咨询&#xff0c;关于基于深度学习和神经网络算法的情绪识别检测系统。回顾往期文章【点击这里】&#xff0c;介绍了关于人脸数据的预处理和模型训练&#xff0c;这里就不在赘述。今天&#xff0c;将详细讲解如何从零基础手写情绪检测算法和情绪检测系统。主…...

本地依赖库的版本和库依赖的版本不一致如何解决?

我用的 yarn v4 版本&#xff0c;所以以下教程命令都基于yarn 这里假设我报错的库名字叫 XXXXXXXX&#xff0c;依赖他的库叫 AAAAAAAA 排查解决思路分析&#xff1a; 首先查看一下 XXXXXXXX 的依赖关系&#xff0c;执行 yarn why XXXXXXXX 首先我们要知道 yarn 自动做了库…...

Redis学习打卡-Day7-高可用(下)

前面提到&#xff0c;在某些场景下&#xff0c;单实例存Redis缓存会存在的几个问题&#xff1a; 写并发&#xff1a;Redis单实例读写分离可以解决读操作的负载均衡&#xff0c;但对于写操作&#xff0c;仍然是全部落在了master节点上面&#xff0c;在海量数据高并发场景&#x…...

Spark on Yarn 高可用模式部署流程

一、引言 Spark是一个用于大规模数据分析处理的分布式计算框架,适用于快速处理大数据的场景。Yarn是一个资源调度框架,用于集群资源的调度和管理。Spark 的任务也可以提交到Yarn中运行,由Yarn进行资源调度。在生产环境中,为了避免单点故障导致整个集群不可用的情况,一个很…...

AI时代新词-大模型(Large Language Model)

一、什么是大模型&#xff1f; 大模型&#xff0c;全称为“大规模语言模型”&#xff08;Large Language Model&#xff09;&#xff0c;是一种基于深度学习的人工智能技术。它通过海量的文本数据进行训练&#xff0c;学习语言的模式、语法和语义&#xff0c;从而能够生成自然…...

3d tiles高级样式设计与条件渲染

条件渲染是3D Tiles样式设置的一大亮点。我们可以通过设置不同的条件来实现复杂的视觉效果。例如&#xff0c;根据建筑物与某个特定点的距离来设置颜色和是否显示&#xff1a; tiles3d.style new Cesium.Cesium3DTileStyle({defines: {distance: "distance(vec2(${featur…...

Linux中logger命令的使用方法详解

文章目录 一、基础语法二、核心功能选项三、‌设施与优先级对照‌1. 常用设施&#xff08;Facility&#xff09;2. 优先级&#xff08;Priority&#xff09;从低到高&#xff1a;3. 组合示例‌ 四、典型使用场景1. 记录简单消息2. 带标签和优先级3. 记录命令输出4. 发送到远程服…...

博奥龙Nanoantibody系列IP专用抗体

货号名称BDAA0260 HRP-Nanoantibody anti Mouse for IP BDAA0261 AbBox Fluor 680-Nanoantibody anti Mouse for IP BDAA0262 AbBox Fluor 800-Nanoantibody anti Mouse for IP ——无轻/重链干扰&#xff0c;更高亲和力和特异性 01Nanoantibody系列抗体 是利用噬菌体展示纳…...

webpack构建速度和打包体积优化方案

一、分析工具 1.1 webpack-bundle-analyzer 生成 stats.json 文件 打包命令webpack --config webpack.config.js --json > stats.json使用 webpack-bundle-analyzer 插件const BundleAnalyzerPlugin = require(webpack-bundle-analyzer).BundleAnalyzerPlugin; plugins: […...

[IMX] 08.RTC 时钟

代码链接&#xff1a;GitHub - maoxiaoxian/imx 目录 1.IMX 的 SNVS 模块 2.SNVS 模块的寄存器 2.1.命令寄存器 - SNVS_HPCOMR 2.2.低功耗控制寄存器 - SNVS_LPCR 2.3.HP 模式的计数寄存器 MSB - SNVS_HPRTCMR 2.4.HP 模式的计数寄存器 LSB - SNVS_HPRTCLR 2.5.LP 模式的…...

PG Craft靶机复现 宏macro攻击

一. 端口扫描 只有80端口开启 二. 网页查看 目录扫描一下&#xff1a; dirsearch -u http://192.168.131.169/ 发现 http://192.168.131.169/upload.php 网站书使用xampp搭建&#xff0c;暴露了路径 还发现上传文件 http://192.168.131.169/uploads/ 发现一个上传点&#x…...

Qt Creator快捷键合集

前言 QtCreator是一款跨平台的IDE,专为Qt开发设计,支持C/C++/JS/Python编程,支持设备远程调试,支持代码高亮,集成帮助文档,原生支持cmake和git,确实是一款朴实而又强大的集成开发环境,让人有种爱不释手的感觉 编辑 功能快捷键复制Ctrl + C粘贴Ctrl + V剪切Ctrl + X代…...

ElasticSearch--DSL查询语句

ElasticSearch DSL查询文档 分类 查询类型功能描述典型应用场景示例语法查询所有匹配所有文档&#xff0c;无过滤条件数据预览/测试json { "query": { "match_all": {} } }全文检索查询对文本字段分词后匹配&#xff0c;基于倒排索引搜索框模糊匹配、多字段…...

海康威视摄像头C#开发指南:从SDK对接到安全增强与高并发优化

一、海康威视SDK核心对接流程​​ 1. ​​开发环境准备​​ ​​官方SDK获取​​&#xff1a;从海康开放平台下载最新版SDK&#xff08;如HCNetSDK.dll、PlayCtrl.dll&#xff09;。​​依赖项安装​​&#xff1a;确保C运行库&#xff08;如vcredist_x86.exe&#xff09;与S…...

Redis(四) - 使用Python操作Redis详解

文章目录 前言一、下载Python插件二、创建项目三、安装 redis 库四、新建python软件包五、键操作六、字符串操作七、列表操作八、集合操作九、哈希表操作十、有序集合操作十一、完整代码1. 完整代码2. 项目下载 前言 本文是基于 Python 操作 Redis 数据库的实战指南&#xff0…...

Kotlin全栈工程师转型路径

针对 Android 开发者向全栈工程师的转型&#xff0c;结合 Kotlin 语言的独特优势&#xff0c;以下是分阶段转型路径和关键技术建议&#xff1a; 一、Kotlin 全栈技术栈构建 后端开发深化 Ktor 框架进阶&#xff1a; 掌握路由嵌套、内容协商&#xff08;JSON/Protobuf&#xf…...

如何利用 Spring Data MongoDB 进行地理位置相关的查询?

以下是如何使用 Spring Data MongoDB 进行地理位置相关查询的步骤和示例&#xff1a; 核心概念&#xff1a; GeoJSON 对象: MongoDB 推荐使用 GeoJSON 格式来存储地理位置数据。Spring Data MongoDB 提供了相应的 GeoJSON 类型&#xff0c;如 GeoJsonPoint, GeoJsonPolygon, …...

服务器并发实现的五种方法

文章目录 前言一、单线程 / 进程二、多进程并发三、多线程并发四、IO多路转接&#xff08;复用&#xff09;select五、IO多路转接&#xff08;复用&#xff09;poll六、IO多路转接&#xff08;复用&#xff09;epoll 前言 关于网络编程相关知识可看我之前写过的文章&#xff1…...

PYTORCH_CUDA_ALLOC_CONF基本原理和具体示例

PYTORCH_CUDA_ALLOC_CONFmax_split_size_mb 是 PyTorch 提供的一项环境变量配置&#xff0c;用于控制 CUDA 显存分配的行为。通过指定此参数&#xff0c;可以有效管理 GPU 显存的碎片化&#xff0c;缓解因显存碎片化而导致的 “CUDA out of memory”&#xff08;显存溢出&#…...

2025年系统架构师---综合知识卷

1.进程是一个具有独立功能的程序关于某数据集合的一次运行活动,是系统进行资源分配和调度的基本单位(线程包含于进程之中,可并发,是系统进行运算调度的最小单位)。一个进程是通过其物理实体被感知的,进程的物理实体又称为进程的静态描述,通常由三部分组成,分别是程序、…...

AI 抠图软件批量处理 + 发丝级精度,婚纱 / 玻璃一键抠透明 免安装

各位抠图小能手们&#xff0c;今天我要给大家介绍一款超厉害的工具——AiartyImageMattingPortable&#xff01;它是基于人工智能的便携式图像抠图工具&#xff0c;专门为快速、精准抠图而生&#xff0c;处理复杂边缘和透明物体那简直就是它的拿手好戏&#xff01; 咱先说说它…...