[C++]vector的模拟实现
下面是简单的实现vector的功能,没有涉及使用内存池等复杂算法来提高效率。
一、vector的概述
(一)、抽象数据类型定义
容器:向量(vector) |
vector是表示大小可以变化的数组的序列容器。像数组一样,向量对其元素使用连续的存储位置,这意味着也可以使用指向其元素的常规指针上的偏移量来访问其元素,并且与数组中的元素一样高效。但与数组不同的是,它们的大小可以动态变化,它们的存储由容器自动处理。 |
模板类 template<class T> 操作 一、构造函数 vector(int n, const T val = T()); 防止vector(int,int)与下面的模板函数发生歧义 vector(std::initializer_list list); 用初始化列表进行初始化 二、迭代器 三、容量 四、访问数据 五、修改操作 |
(二)、成员变量
template<class T>
class vector
{
public://vector的迭代器可以为原生指针typedef T* iterator;typedef const T* const_iterator;
private:iterator _start = nullptr;//指向第一个数据的迭代器iterator _finish = nullptr;//指向最后一个有效数据的后一个位置iterator _endOfStorage = nullptr;//指向数组末尾的后一个位置
};
这里的迭代器就是我们的原生指针T*,它的私有成员变量就和线性表一样,要存储数据的起始位置的指针、数据的有效数据的个数、存储容量等。这里直接全部定义成指针来维护容器的数据数组,_start指向数组的起始位置,_finish指向有效数据个数的后一个的位置,当我们将这两个指针相减,_finish - _start 就能得到数据的有效数据个数了。最后一个endOfStorage就是指向数组的末尾的下一个位置(虽然越界但不解引用就没事),当我们拿_endOfStorage - _start 就得到了这段空间的容量个数。
注意:我们的每个构造函数,在一开始时都要把迭代器置为空指针,而我们在声明迭代器时,顺便给上初始值,这样会更好,以免忘记初始化带来不必要的麻烦。
迭代器维护的空间示意图:
二、具体实现
我们不按上面的函数声明的顺序来依次定义函数,我们按照从易到难的顺序,一点一点实现其成员函数的定义,每个阶段测试一下。
(一)、默认构造、析构、迭代器、尾插、尾删、下标访问
(1)、默认构造、析构
//默认构造
vector() :_start(nullptr), _finish(nullptr), _endOfStorage(nullptr) { ; }
因为是空容器,所以三个指针都初始化为nullptr。
//析构
~vector()
{delete[] _start;_start = _finish = _endOfStorage = nullptr;
}
删除动态分配的空间后记得将指针置为空,因为析构是可以手动调用的。
(2)、迭代器
//迭代器
iterator begin() { return _start; }
iterator end() { return _finish; }
const_iterator begin() const { return _start; }
const_iterator end() const { return _finish; }
两组迭代器分别构成重载。
注意:重载不是因为返回值的不同,而是其参数不同,一个是this,另一个是const this的指针。
(3)、尾插和尾删操作
注意:需要对容器满时进行扩容处理。
1、辅助其扩容的函数
empty():判断容器是否为空
bool empty() const { return (_finish - _start) == 0; }
size():返回容器的有效数据个数
size_t size() const { return _finish - _start; }
capacity():返回容器的容量
size_t capacity() const { return _endOfStorage - _start; }
reserve():扩容操作
void reserve(size_t n)//改变容量大小
{//如果n为0,默认给4个空间n = n == 0 ? 4 : n;//不允许缩容if (n <= capacity())return;//扩容size_t nSize = size();//保存有效数据个数iterator itTem = new T[n];//拷贝到新数组for (int i = 0; i < nSize; i++){*(itTem + i) = *(_start + i);}delete[] _start;_start = itTem;_finish = _start + nSize;_endOfStorage = _start + n;
}
2、尾插、尾删
//修改操作
void push_back(const T& val)//尾插操作
{//判断容器空间是否足够if (size() == capacity()){size_t newCapacity = capacity() * 2;//两倍扩容//扩容reserve(newCapacity);//std::cout << capacity() << " ";}//尾插*_finish = val;_finish++;
}
void pop_back()//尾删操作
{if (!empty())_finish--;
}
(4)、下标访问
//访问操作T& operator[](size_t n){//防止越界访问assert(n >= 0 && n < size());return *(_start + n);}const T& operator[](size_t n) const{//防止越界访问assert(n >= 0 && n < size());return *(_start + n);}
注意:构成函数重载的是参数,不是返回值。
(5)、代码测试
void test1()
{const int N = 10;//测试数据的个数MySpace::vector<int> v1;for (int i = 0; i < N; i++){v1.push_back(i + 1);}//手动使用迭代器遍历MySpace::vector<int>::iterator it1 = v1.begin();while (it1 != v1.end()){cout << *it1 << " ";it1++;}cout << endl;//数组方式访问for (int i = 0; i < N; i++)cout << v1[i] << " ";cout << endl;//测试尾删for (int i = 0; i < N + 2; i++)//+2是为了测试一下删空容器{cout << endl;for (auto e : v1)//测试范围for{cout << e << " ";}v1.pop_back();}
}
运行结果:
1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9
1 2 3 4 5 6 7 8
1 2 3 4 5 6 7
1 2 3 4 5 6
1 2 3 4 5
1 2 3 4
1 2 3
1 2
1
(二)、其余的构造函数
(1)、迭代器区间和参数初始化表初始化
迭代器区间初始化构造函数
因为迭代器的类型可能不同,不一定所有的迭代器的类型都是指针,有可能是自定义类型。所以用模板函数。注意:迭代器不一定都支持加减操作,所以用迭代器加减的方式获得迭代器区间的大小是不安全的做法。
//迭代器区间初始化
template<class InputIterator>
vector(InputIterator first, InputIterator last)
{//这里不可以像指针那样直接 last - first 算出数据个数//因为有的迭代器是不支持加减的//可以采取以下方法先算大小,防止频繁扩容导致效率降低size_t n = 0;InputIterator itFirst = first;InputIterator itLast = last;while (itFirst != itLast){n++;itFirst++;}reserve(n);while (first != last){push_back(*first);first++;}
}
参数初始化列表初始化构造函数
//用参数初始化表初始化
vector(std::initializer_list<T> list)
{ reserve(list.size());for (auto& e: list){push_back(e);}
}
(2)拷贝构造函数
//拷贝构造vector(const vector<T>& v){reserve(v.capacity());for (auto& e : v){push_back(e);}}
(3)交换两个容器和赋值构造函数
实现交换容器的函数swap()来辅助赋值构造函数的实现
void swap(vector<T>& v)//交换两个的数据
{std::swap(_start, v._start);std::swap(_finish,v._finish);std::swap(_endOfStorage, v._endOfStorage);
}
赋值构造函数
//赋值构造
vector<T>& operator=(const vector<T>& v)
{//复用拷贝构造函数vector<T> tem(v);swap(tem);return *this;
}
(4)填充初始化
//填充初始化vector(size_t n, const T val = T())//用n个val进行初始化{reserve(n);for (int i = 0; i < n; i++){push_back(val);}}//用来防止调用vector(int,int)时发生歧义//发生歧义的函数:vector<InputIterator InputIterator>vector(int n, const T val = T()){reserve(n);for (int i = 0; i < n; i++){push_back(val);}}
(5)、代码测试
void test2()
{const int N = 8;MySpace::vector<int> v1;for (int i = 0; i < N; i++){v1.push_back(i + 1);}for (auto e : v1){cout << e << " ";}cout << endl;//测试迭代器区间初始化构造MySpace::vector<int> v2(v1.begin() + 2, v1.end()-2);for (auto e : v2){cout << e << " ";}//测试参数初始化列表构造cout << endl;MySpace::vector<int> v3({ 1,2,3,4,5,6,7,8,9,10 });for (auto e : v3){cout << e << " ";}//测试拷贝构造cout << endl;MySpace::vector<int> v4(v3);for (auto e : v4){cout << e << " ";}//测试赋值构造cout << endl;MySpace::vector<int> v5;v5 = v4;for (auto e : v5){cout << e << " ";}//测试填充初始化cout << endl;MySpace::vector<int> v6(10,7);for (auto e : v6){cout << e << " ";}
}
运行结果:
1 2 3 4 5 6 7 8
3 4 5 6
1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9 10
7 7 7 7 7 7 7 7 7 7
(三)、指定位置插入和删除
(1)、指定位置插入
iterator insert(iterator pos, const T& x)//在指定位置插入{//判断插入的位置及是否合法assert(pos >= _start && pos <= _finish);//判断是否需要扩容if (size() == capacity()){//防止扩容后pos迭代器失效size_t nPos = pos - _start;//计算pos距离_start的偏移量size_t newCapacity = capacity() * 2;reserve(newCapacity);pos = _start + nPos;}//将插入位置及之后的元素往后挪iterator it = _finish;while (it > pos){*it = *(it - 1);it--;}//插入元素*pos = x;_finish++;return pos;}
注意:扩容时pos迭代器会失效,扩容前要计算pos距离起始位置的偏移量,然后再更新pos的迭代器。所以,外部的迭代器pos是有失效的可能性的,而不管怎么样外部的pos是不能再继续使用的,若要继续使用,则需要接受函数的返回值更新pos的迭代器,防止迭代器失效,此时pos迭代器指向新插入的数据。
(2)、指定位置删除
iterator erase(iterator pos)//指定位置删除
{//判断删除的位置是否合法assert(pos >= _start && pos <= _finish - 1);//将pos位置后的元素往前挪iterator cur = pos + 1;while (cur != _finish){*(cur - 1) = *cur;cur++;}_finish--;return pos;
}
注意:vector删除元素时有可能缩容的,虽然这里实现的vector不会缩容,为了防止外部的迭代器失效,我们也需要返回一个迭代器更新其pos的位置,此时pos指向的位置是删除的元素的下一个元素的位置。
(3)、代码测试
void test3()
{MySpace::vector<int> v1{ 1,2,3,4,5,6 };//测试指定位置插入v1.insert(v1.begin(), 999);v1.insert(v1.end(), 999);v1.insert(v1.end() - 2, 999);for (auto e : v1){cout << e << " ";}cout << endl;//结合find删除所有999//找不到则返回结尾的迭代器MySpace::vector<int>::iterator itFind;while ((itFind = std::find(v1.begin(), v1.end(), 999)) != v1.end()){cout << endl;v1.erase(itFind);for (auto e : v1){cout << e << " ";}}
}
(4) 运行结果:
999 1 2 3 4 5 999 6 999
1 2 3 4 5 999 6 999
1 2 3 4 5 6 999
1 2 3 4 5 6
(四)、修改有效数据个数大小
具体实现代码:
void resize(size_t n)//改变容器数据个数
{assert(n >= 0);//判断是否需要扩容if (n > capacity()){reserve(n);}//将其他位置初始化为数据的默认构造for (int i = size(); i < n; i++){*(_start + i) = T();}_finish = _start + n;
}
代码测试:
void test4()
{//测试resizeMySpace::vector<int> v1{ 1,2,3,4,5,6,7,8,9,10 };for (auto e : v1){cout << e << " ";}cout << endl;//缩容v1.resize(2);for (auto e : v1){cout << e << " ";}cout << endl;//调大有效数据个数大小v1.resize(10);for (auto e : v1){cout << e << " ";}
}
1 2 3 4 5 6 7 8 9 10
1 2
1 2 0 0 0 0 0 0 0 0
以上就是vector的简易实现。
相关文章:
[C++]vector的模拟实现
下面是简单的实现vector的功能,没有涉及使用内存池等复杂算法来提高效率。 一、vector的概述 (一)、抽象数据类型定义 容器:向量(vector)vector是表示大小可以变化的数组的序列容器。像数组一样…...
【云原生】Kubernetes----POD控制器
目录 引言 一、Pod控制器概述 二、Pod控制器的种类 (一)ReplicaSet (二)Deployment (三)StatefulSet (四)DaemonSet (五)Job 三、使用POD控制器 &a…...
Java环境配置(超详细)
Java环境配置(超详细) 引言1、安装 JDK1.1、下载安装JDK1.2、配置环境变量:JAVA_HOME1.3、将JAVA_HOME添加到Path中 2、安装 Maven2.1、下载安装Maven2.2、配置maven的环境变量: M2_HOME2.3、将Maven变量添加到Path中 引言 Java开发环境的配…...
【操作系统】(详细理解进程的状态)执行状态、就绪状态、阻塞状态、挂起状态
下面是进程的几种状态的概念: 执行状态:当一个进程已获得必要资源,并占有CPU进行执行。 就绪状体:进程已分配到除CPU外的所有必要资源,只要获取CPU允许就可立即执行。 阻塞状态:正在执行的进程,…...
C++ -- string常用接口的底层实现
一.string介绍 1. string是表示字符串的字符串类,对C语言的字符串指针进行了包装。 2. 该类的接口与常规容器的接口基本相同,有增删查改等,再添加了一些专门用来操作string的常规操作。 二.成员变量 创建string类的时候要在自己的命名空间…...
怎么做好企业短信服务呢?(文字短信XML接口示例)
企业短信服务已经成为各行各业都信赖的行业推广方式之一,并且短信行业也与时俱进的发展着,随之而来的就是市场上短信平台的数量也随之增多。那么怎么在鱼龙混杂的短信行业中选择适合自己的企业短信服务平台呢?企业短信服务平台又适用于哪些应…...
鸿蒙小案例-音乐播放器
之前参加鸿蒙比赛的音乐播放器 效果展示 HF音乐效果展示 功能列 有一些功能没写上去,自行发掘 说明: 1.API:网易云接口,QQ个人接口, 需要请看gitee 2.本地关系型数据由bug,提的工单已确认,建议使用API11,12,9的不稳…...
语言模型测试系列【9】
语言模型 文心一言讯飞星火通义千问2.5豆包360智脑百小应腾讯元宝KimiC知道 好长时间没有做语言模型的测试了,一方面是没有好的素材,各模型都在升级优化,而且频率很高;另一方面近期在阅读和学习其他的知识,所以更的也…...
优思学院|质量工程师工资不高怎么办?
你是否曾经好奇,为什么在职场中,质量工程师的工资普遍不高?这一现象背后的原因,实际上与他们的职业门槛和专业知识密切相关。早期,国内的质量工程师入行门槛较低,许多人即使没有任何专业知识也可以进入这一…...
【面向就业的Liux基础】从入门到熟练,探索Linux的秘密(一)
主要帮助大家面向工作过程中Linux系统常用的命令联系,采用极致的实用主义,帮助大家节省时间。 文章目录 前言 一、linux系统 二、linux系统基本命令 1.Linux系统的目录结构 2. 常用命令介绍 3.命令演示 4.作业练习 总结 前言 主要帮助大家面向工作过程中…...
高效数据处理的前沿:【C++】、【Redis】、【人工智能】与【大数据】的深度整合
目录 1.为什么选择 C 和 Redis? 2.人工智能与大数据的背景 1.大数据的挑战 2.人工智能的需求 3.C 与 Redis 的完美结合 1.安装 Redis 和 Redis C 客户端 2.连接 Redis 并进行数据操作 高级数据操作 列表操作 哈希操作 4.与大数据和人工智能结合 5.实际应…...
Vitis HLS 学习笔记--控制驱动与数据驱动混合编程
目录 1. 简介 2. 示例分析 2.1 代码分析 2.2 控制驱动TLP的关键特征 2.3 数据驱动TLP的关键特征 3. 总结 1. 简介 在 HLS 硬件加速领域,Vitis HLS 提供了强大的抽象并行编程模型。这些模型包括控制驱动和数据驱动的任务级并行性(TLP)&…...
VUE3 学习笔记(12):对比Vuex与Pinia状态管理的基本理解
在组件传值中,当嵌套关系越来越复杂的时候必然会将混乱,是否可以把一些值存在一个公共位置,无须传值直接调用呢?VUEX应运而生,但是从VUE3开始对VUEX的支持就不那么高了,官方推荐使用Pinia。 Vuex配置 ST1:…...
区间预测 | Matlab实现QRCNN-BiGRU-Attention分位数回归卷积双向门控循环单元注意力机制时序区间预测
区间预测 | Matlab实现QRCNN-BiGRU-Attention分位数回归卷积双向门控循环单元注意力机制时序区间预测 目录 区间预测 | Matlab实现QRCNN-BiGRU-Attention分位数回归卷积双向门控循环单元注意力机制时序区间预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.Matlab实…...
TypeScript算法每日一题:赎金信(383)
作者:前端小王hs 阿里云社区博客专家/清华大学出版社签约作者✍/CSDN百万访问博主/B站千粉前端up主 题库:力扣 题目序号:383(简单) 题目:赎金信 给你两个字符串ransomNote 和 magazine,判断ran…...
springboot 作为客户端接收服务端的 tcp 长连接数据,并实现自定义结束符,解决 粘包 半包 问题
博主最近的项目对接了部分硬件设备,其中有的设备只支持tcp长连接方式传输数据,博主项目系统平台作为客户端发起tcp请求到设备,设备接收到请求后作为服务端保持连接并持续发送数据到系统平台。 1.依赖引入 连接使用了netty,如果项…...
kuka编程怎么加中文:解锁KUKA机器人编程中的中文支持
kuka编程怎么加中文:解锁KUKA机器人编程中的中文支持 在工业自动化领域,KUKA机器人以其卓越的性能和广泛的应用而备受赞誉。然而,对于许多中国用户来说,如何在KUKA编程中加入中文支持却成为了一个挑战。本文将从四个方面、五个方…...
hadoop集群中zookeeper的搭建与原理解释
搭建zookeeper 将zookeeper的apache-zookeeper-3.5.7-bin.tar.gz解压到/export/servers下 tar -zxvf apache-zookeeper-3.5.7-bin.tar.gz -C /export/servers为了方便后期使用解压后的文件夹改名为zookeeper-3.5.7 mv apache-zookeeper-3.5.7-bin zookeeper-3.5.7先进入zoo_…...
HTML静态网页成品作业(HTML+CSS)—— 父亲节节日介绍网页(4个页面)
🎉不定期分享源码,关注不丢失哦 文章目录 一、作品介绍二、作品演示三、代码目录四、网站代码HTML部分代码 五、源码获取 一、作品介绍 🏷️本套采用HTMLCSS,未使用Javacsript代码,共有4个页面。 二、作品演示 三、代…...
Client ID 与Client Secret
什么是 Client ID 和 Client Secret? 在现代应用程序中,特别是在涉及到OAuth 2.0身份验证和授权时,Client ID 和 Client Secret是非常重要的概念。它们通常用于验证和授权第三方应用程序,以便安全地访问受保护的资源或API。 Cli…...
React中实现大模型的打字机效果
React 想实现一个打字机的效果,类似千问、Kimi 返回的效果。调用大模型时,模型的回答通常是流式输出的,如果等到模型所有的回答全部完成之后再展示给最终用户,交互效果不好,因为模型计算推理时间比较长。本文将采用原生…...
十二、配置注解执行SQL
简化一下流程,主要可以分为下面几步: 1.解析配置,写入配置项 2.执行SQL 3.封装结果 通过注解配置SQL主要体现在解析部分,这部分要分别做解析XML还是配置注解的接口,拿到sql以后,select的处理和insert/upda…...
阿里云计算之运维概念学习笔记(一)
运维管理 运维管理(Operation and Maintenance Management, 简称O&M管理)是指通过科学的管理方法和技术手段,对IT系统和基础设施进行监控、维护、优化和保障,以确保系统的高可用性、稳定性、安全性和性能。运维管理涵盖了硬件…...
异常概述
自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 在程序运行过程中,经常会遇到各种各样的错误,这些错误统称为“异常”。这些异常有的是由于开发者将关键字敲错导致的…...
【Postman接口测试】第五节.Postman接口测试项目实战(下)
文章目录 前言七、课程添加接口postman测试 7.1 课程添加接口文档 7.2 针对课程添加设计接口测试用例 7.2.1 提取测试点 7.2.2 设计测试用例 7.2.2 使用Postman进行接口测试八、查询课程列表接口postman测试 8.1 查询…...
医用腕带朔源用的条形码与二维码如何选择
在医疗环境中的医用腕带作为患者身份识别和管理的重要工具,做为条形码和二维码腕带上的溯源技术,更是为患者信息快速获取、准确传递的保障,实现更加高效和准确的患者身份识别和管理,这种技术可以大大提高医疗服务的效率和质量&…...
“Kubectl 如何工作案例:编写自定义 Kubectl 命令
Kubernetes 工作起来就像魔法,但它并不是魔法。它本质上是基于 REST API 调用的简单性。这种直截了当的机制是其强大功能的关键。今天,我们将深入探讨 Kubernetes 的内部工作原理,特别是当我们执行 kubectl 命令时幕后发生了什么。 1.1 AUTHENTICATION 默认情况下,kubect…...
opencv-python(五)
opencv的颜色通道中顺序是B,G,R。 图像属性 import cv2img cv2.imread(jk.jpg) print(fshape{img.shape}) print(fsize{img.size}) print(fdtype{img.dtype}) shape:图像像素的行,列,通道 size:行数 X …...
免费生物蛋白质的类chatgpt工具助手copilot:小分子、蛋白的折叠、对接等
参考: https://310.ai/copilot 可以通过自然语言对话形式实现小分子、蛋白质的相关处理:生成序列、折叠等 应该是agent技术调用不同工具实现 从UniProt数据库中搜索和加载蛋白质。使用ESM Fold方法折叠蛋白质。使用310.ai基础模型设计新蛋白质。使用TM-Align方法比较蛋白质…...
Mybatis01-初识Mybatis
简介 1、 什么是Mybatis MyBatis 是一款优秀的持久层框架; 它支持自定义 SQL、存储过程以及高级映射 MyBatis 免除了几乎所有的 JDBC 代码以及设置参数和获取结果集的工作。 MyBatis 可以通过简单的 XML 或注解来配置和映射原始类型、接口和 Java POJO(Plain Ol…...
中国服务外包研究中心/seo一个月赚多少钱
亚洲草原上有一种野马,长鬃飘飘,骏逸而健壮。那里同时生活着一种吸血蝙蝠,被称作野马的天敌。这种个头不大的吸血鬼非常难缠,一旦叮上野马,便如沥青粘身,非得吸饱喝足方肯离去。如果光是吸血,而…...
h5自助建站系统/郑州网站建设优化
北京时间2017年6月14日,为期三天的第九届中国云计算大会在北京国家会议中心拉开序幕。本届大会以“生态构建 深化应用”为主题,重点围绕云计算大数据的产业生态完善、技术与传统产业的深度融合,以及云计算大数据技术与物联网、人工智能等最新…...
哈尔滨网站建设效果/千万不要学网络营销
想打开页面自动定位到输入框并弹出输入法,试了很多方法都不行,后来看到下面这篇文章,安装他分析的思路,可能是要等view绘制完成了弹出输入法才有效,所以需要延时弹出输入法,试了,确实有效 http…...
建站平台隐藏技术支持/上海市人大常委会
使用url传递中文参数时,先使用js自带的转码工具进行转码: sysMselExp.jsp?MEETINGNAME"meeting"&EXPGROUP"encodeURIComponent(group)"&MARK1";//此处为group转码 然后再页面头部取值时,采用如下方式&…...
佛山八戒网站建设/百度seo营销推广
第1篇:《Thereare…》教学反思本课的教学内容是第五册module5unit1的第一课时,学生能够理解并运用句型thereare…,thereareenough/therearen’tenough是本课的教学目标,通过本课的学习,学生们不仅理解了本课的教学目标…...
志勋网站建设公司/建网站教学
变量, 常量, 类型 import org.example.Main import java.util.stream.Stream// 编译时常量, 支持如下类型 : String ,Int ,Double ,Float ,Long ,Short ,Byte ,Char ,Booleanconst val MAX_VAlUE 256 fun main() {// 声明一个变量var aInt: Int 1 // kotlin可以自动推断出类…...