C++vector模拟实现

vector模拟实现

  • 1.构造函数
  • 2.拷贝构造
  • 3.析构+赋值运算符重载
  • 4.iterator
  • 5.modifiers
    • 5.1push_back
    • 5.2pop_back
    • 5.3empty
    • 5.4insert
    • 5.5erase
    • 5.6swap
  • 6.Capacity
    • 6.1size
    • 6.2capacity
    • 6.3reserve
    • 6.4resize
    • 6.5empty
  • 7.Element access
    • 7.1operator[]
    • 7.2at
  • 8.在谈reserve

vector官方库实现的是一个模板类,因此我们也写个模板类。
在前面模拟实现过string类,string和vector都是动态增长的数组,有点类似。

在这里插入图片描述
vector把这个三个成员变量都换成了指针。

template<class T>
class Vector
{
public:typedef T* iterator;typedef const T* const_iterator;private:iterator _start;iterator _finish;iterator _endofstorage;
};

1.构造函数

在这里插入图片描述

	//第一种无参构造Vector():_start(nullptr),_finish(nullptr),_endofstorage(nullptr){}
	//第二种构造Vector(size_t n, const T& val = T()):_start(nullptr), _finish(nullptr), _endofstorage(nullptr){//扩容if (_finish == _endofstorage){//因为这里没有size,capacity所以自己写个size,capacityint newcapacity = capacity() == 0 ? 4 : 2 * capacity();reserve(newcapacity);}while (n--){push_back(val);}}void reserve(size_t n){if (n > capacity()){T* tmp = new T[n];//_start不为nullptr再拷贝if (_start){memcpy(tmp, _start, sizeof(T) * size());delete[] _start;}_start = tmp;_finish = _start + size();_endofstorage = _start + n;}}size_t size() const{//指针减指针等于元素个数return _finish - _start;}size_t capacity() const{return _endofstorage - _start;}void push_back(const T& val = T()){//扩容if (_finish == _endofstorage){//因为这里没有size,capacity所以自己写个size,capacityint newcapacity = capacity() == 0 ? 4 : 2 * capacity();reserve(newcapacity);}*_finish = val;++_finish;}

写完一段代码先测试测试,看看有没有错误。
在这里插入图片描述
这里的_finish报错了,调试一下找原因。如果不能确定到底是哪里错误,就一段一段屏蔽代码,这里我们最终确定是扩容出现了问题。
在这里插入图片描述
我们开辟空间之后,_finish还指向nullptr,也就是根本没有更新_finish的位置。

	void reserve(size_t n){if (n > capacity()){//记录_finish的相对位置int Oldsize = size();T* tmp = new T[n];//_start不为nullptr再拷贝if (_start){memcpy(tmp, _start, sizeof(T) * size());delete[] _start;}_start = tmp;//对_finish特殊处理。//_finish=_start+_finish-_start导致的错误。//_finish = _start + size();_finish = _start + Oldsize;_endofstorage = _start + n;}}

这样写,也防止了_start本身有空间,然后异地扩容,_finish位置不对的情况。

这里的reserve内部memcpy还有一个浅拷贝的问题,等最后说到这个成员函数再具体谈。

	//第三种扩容template <class InputIterator>Vector(InputIterator first, InputIterator last):_start(nullptr),_finish(nullptr),_endofstorage(nullptr){while (first != last){push_back(*first);++first;}}iterator begin() const{return _start;}iterator end() const{return _finish;}

在这里插入图片描述

注意看图片的对比,右边没有屏蔽第二种构造函数,竟然报错了。
在这里插入图片描述
把1换成字符‘c’就不报错了。

其原因是因为10,1都是int类型,走的是模板参数的构造。
不是走的Vector(size_t n, const T& val = T()),因为int转换成size_t涉及算术转换。
而Vector(InputIterator first, InputIterator last) 这里都是同一类型,所以编译器会选择模板。
这里和库的解决方法一样,在加一个Vector(int n, const T& val = T())。
这样由于由现成的,编译器就不再会走模板了。

Vector(int n, const T& val = T()):_start(nullptr), _finish(nullptr), _endofstorage(nullptr){//扩容if (_finish == _endofstorage){//因为这里没有size,capacity所以自己写个size,capacityint newcapacity = capacity() == 0 ? 4 : 2 * capacity();reserve(newcapacity);}while (n--){push_back(val);}}

在这里插入图片描述

2.拷贝构造

在string模拟实现,拷贝构造实现了现代写法,是比较方便的,我们这里就继续用起来。

void swap(Vector<T> x){//这里调用的是库里面swap,关于这里可以看看string类模拟实现std::swap(_start, x._start);std::swap(_finish, x._finish);std::swap(_endofstorage, x._endofstorage);}Vector(const Vector<T>& val):_start(nullptr),_finish(nullptr),_endofstorage(nullptr){Vector<T> tmp(val.begin(), val.end());//这里调用的是自己写的swap,如果直接用库里面的就会调用构造和拷贝构造。swap(tmp);}

3.析构+赋值运算符重载

	~Vector(){delete[] _start;_start = _finish = _endofstorage = nullptr;}//赋值运算符重载现代写法//v1=v2//v1=v1  虽然这里没有判断,但是很少有人会自己给自己赋值//即使赋值了,也是使用了一次拷贝构造Vector<T>& operator=(Vector<T> val){swap(val);return *this;}

4.iterator

	iterator begin() const{return _start;}const_iterator begin() const{return _start;}iterator end() const{return _finish;}const_iterator end() const{return _finish;}

5.modifiers

5.1push_back

void push_back(const T& val = T()){//扩容if (_finish == _endofstorage){//因为这里没有size,capacity所以自己写个size,capacityint newcapacity = capacity() == 0 ? 4 : 2 * capacity();reserve(newcapacity);}*_finish = val;++_finish;}

5.2pop_back

	void pop_push(){assert(!empty());--_finish;}

5.3empty

	bool empty(){return _start == _finish;}

5.4insert

在这里插入图片描述

	void insert(iterator pos, const T& val){assert(pos >= _start);assert(pos < _finish);//扩容if (_finish == _endofstorage){int newcapacity = capacity() == 0 ? 4 : 2 * capacity();reserve(newcapacity);}iterator begin = end();while (begin > pos){*begin = *(begin - 1);--begin;}*pos = val;++_finish;}

这样写就没有问题了吗?
测试一下看看。
在这里插入图片描述
发现和我们预想的结果不一样,分析一下,找原因
在这里插入图片描述
请问pos,在扩容之后应该在哪里呢?

这里是迭代器失效:野指针问题。
扩容导致,插入pos位置还是原空间的位置。
解决方法:
更新pos的位置。

	void insert(iterator pos, const T& val){assert(pos >= _start);assert(pos < _finish);//扩容if (_finish == _endofstorage){//记录pos的相对位置size_t n = pos - _start;int newcapacity = capacity() == 0 ? 4 : 2 * capacity();reserve(newcapacity);//更新pospos = _start + n;}iterator begin = end();while (begin > pos){*begin = *(begin - 1);--begin;}*pos = val;++_finish;}

5.5erase

	void erase(iterator pos){assert(pos >= _start)assert(pos < _finish);iterator begin = pos + 1;while (begin < _finish){*(begin - 1) = *begin;++begin;}--_finish;}

在来测试一下。

在这里插入图片描述
在这里插入图片描述

发现自己实现的vector没报错,库里面的vector,删除it位置,然后再读it位置,就报错了。这是也是因为迭代器失效问题

再看看linux环境下同样代码是上面情况。
在这里插入图片描述
在这里插入图片描述
linux环境下读it位置可以再次读,说明这个位置没有失效。

那么erase删除位置到底失效不失效呢?

erase删除it位置,失效不失效,由编译器自己决定。

再看这样一种情况。
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
两个平台出现不一样的结果,为了代码能够再不同平台都能正常运行。所以,

erase,删除it位置元素,统一认为it失效了。it就不能再读写访问了,更新it才能继续访问,所以erase返回被删元素的下一个位置

在这里插入图片描述
更新了lt位置,就不会报错了。

正确写法。

	iterator erase(iterator pos){assert(pos < _finish);iterator begin = pos + 1;while (begin < _finish){*(begin - 1) = *begin;++begin;}--_finish;return pos;}

5.6swap

	void swap(Vector<T>& x){std::swap(_start, x._start);std::swap(_finish, x._finish);std::swap(_endofstorage, x._endofstorage);}

6.Capacity

6.1size

	size_t size() const{//指针减指针等于元素个数return _finish - _start;}

6.2capacity

	size_t capacity() const{return _endofstorage - _start;}

6.3reserve

void reserve(size_t n){if (n > capacity()){//记录_finish的相对位置int Oldsize = size();T* tmp = new T[n];//_start不为nullptr再拷贝if (_start){memcpy(tmp, _start, sizeof(T) * size());delete[] _start;}_start = tmp;//对_finish特殊处理。//_finish=_start+_finish-_start导致的错误。//_finish = _start + size();_finish = _start + Oldsize;_endofstorage = _start + n;}}

6.4resize

resize有三种情况;
n<size();删除元素
size()<n<capacity();插入元素
n>capacity();扩容+插入元素

void resize(size_t n, const T& val=T()){if (n > size()){//这里在reserve已经判断过了是否扩容reserve(n);//插入数据while (_finish < _start + n){*_finish = val;++_finish;}}else{_finish = _start + n;}}

6.5empty

	bool empty(){return _start == _finish;}

7.Element access

7.1operator[]

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

7.2at

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

剩下接口比较简单,这里就不再模拟实现了。

8.在谈reserve

在谈reserve之前,做一道LeetCode的题
118. 杨辉三角

在这里插入图片描述

class Solution {
public:vector<vector<int>> generate(int numRows) {vector<vector<int>> vv;vv.resize(numRows);for(int i=0;i<vv.size();++i){vv[i].resize(i+1);//放数据for(int j=0;j<vv[i].size();++j){if(j == 0 || j ==  vv[i].size()-1){vv[i][j]=1;}else{vv[i][j]=vv[i-1][j-1]+vv[i-1][j];}}}return vv;}
};

注意这道题,vector < T > ,T是自定义类型,不是内置类型;我们用自己写的vector来跑一下类似的代码,看看有没有什么问题。

void test_Vector4()
{Vector<Vector<int>> vv;Vector<int> v(5, 1);vv.push_back(v);vv.push_back(v);vv.push_back(v);vv.push_back(v);vv.push_back(v);for (size_t i = 0; i < vv.size(); ++i){for (size_t j = 0; j < vv[i].size(); ++j){cout << vv[i][j] << " ";}cout << endl;}}

在这里插入图片描述为什么会有随机值的出现呢?
画图分析。
在这里插入图片描述
原因在于,reserve中memcpy拷贝的时候是浅拷贝,delete[ ] _start会把原有空间给释放掉了,所以出现随机值问题,

解决方法:浅拷贝换成深拷贝。

void reserve(size_t n){if (n > capacity()){//记录_finish的相对位置int Oldsize = size();T* tmp = new T[n];//_start不为nullptr再拷贝if (_start){//浅拷贝//memcpy(tmp, _start, sizeof(T) * size());//深拷贝for (size_t i = 0; i < Oldsize; ++i){tmp[i] = _start[i];}delete[] _start;}_start = tmp;//对_finish特殊处理。//_finish=_start+_finish-_start导致的错误。//_finish = _start + size();_finish = _start + Oldsize;_endofstorage = _start + n;}}

自此关于vector类模拟实现结束了。里面有很多值得多多思考的东西,希望能给你带来一份收获,喜欢的小伙伴,点赞,评论,收藏+关注!下篇博文再见!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/77486.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SQL11 高级操作符练习(1)

描述 题目&#xff1a;现在运营想要找到男性且GPA在3.5以上(不包括3.5)的用户进行调研&#xff0c;请你取出相关数据。 示例&#xff1a;user_profile iddevice_idgenderageuniversitygpa12138male21北京大学3.423214male复旦大学4.036543female20北京大学3.242315female23浙…

向量范数及其Python代码

【向量范数】 向量由于既有大小又有方向&#xff0c;所以不能直接比较大小。 向量范数通过将向量转化为实数&#xff0c;然后进行向量的大小比较。 所以&#xff0c;向量范数是用于度量“向量大小”的量。 设向量 &#xff0c;则有&#xff1a; ● 向量的 范数&#xff1a; ●…

Python计算机Python二级知识点整理

1. 此时我们这里首先解析一下这个d[A]N,根据ASCII表&#xff0c;我们可以看出字符A对应的十进制数字是65&#xff0c;ord()函数是把字符转换为相对应的ASCII码&#xff0c;chr()函数是ord()函数的逆运算&#xff0c;所以ord("A")65 ,chr(65)A,题目中首先定义了d为一…

C++毕业设计基于QT实现的超市收银管理系统源代码+数据库

C毕业设计基于QT实现的超市收银管理系统源代码数据库 编译使用 编译完成后&#xff0c;需要拷贝 file目录下的数据库 POP.db文件到可执行程序目录下 登录界面 主界面 会员管理 完整代码下载地址&#xff1a;基于QT实现的超市收银管理系统源代码数据库

笔记本多拓展出一个屏幕

一、首先要知道&#xff0c;自己的电脑有没有Type-c接口&#xff0c;支持不支持VGA 推荐&#xff1a; 自己不清楚&#xff0c;问客服&#xff0c;勤问。 二、显示屏与笔记本相连&#xff0c;通过VGA 三、连接好了&#xff0c;需要去配置 网址&#xff1a;凑合着看&#xff…

LLM 02-大模型的能力

LLM 02-大模型的能力 我们将深入探讨GPT-3——这个具有代表性的大型语言模型的能力。我们的研究主要基于GPT-3论文中的基准测试&#xff0c;这些测试包括&#xff1a; 标准的自然语言处理&#xff08;NLP&#xff09;基准测试&#xff0c;例如问题回答&#xff1b;一些特殊的一…

【OpenCV • c++】图像噪音 | 椒盐噪音 | 高斯噪音

文章目录 一、什么是图像噪音二、椒盐噪声三、高斯噪声 一、什么是图像噪音 图像噪声是图像在获取或是传输过程中受到随机信号干扰&#xff0c;妨碍人们对图像理解及分析处理的信号。很多时候将图像噪声看做多维随机过程&#xff0c;因而描述噪声的方法完全可以借用随机过程的描…

PyCharm中使用matplotlib.pyplot.show()报错MatplotlibDeprecationWarning的解决方案

其实这只是一个警告&#xff0c;忽略也可。 一、控制台输出 MatplotlibDeprecationWarning: Support for FigureCanvases without a required_interactive_framework attribute was deprecated in Matplotlib 3.6 and will be removed two minor releases later. MatplotlibD…

Java“牵手”微店商品列表页数据采集+微店商品价格数据排序,微店API接口申请指南

微店平台创立于2011年5月&#xff0c;是北京口袋时尚科技开发的应用&#xff0c;2014年1月"微店"APP正式上线。微店已经从小微店主首选的开店工具转型为助力创业者发展兴趣、创立品牌、玩成事业的系统及基础设施。 微店商品列表数据包含商品名称、价格、销量、详情、…

微信小程序 通过 pageScrollTo 滚动到界面指定位置

我们可以先创建一个page 注意 一定要在page中使用 因为pageScrollTo控制的是页面滚动 你在组件里用 他就失效了 我们先来看一个案例 wxml 代码如下 <view><button bindtap"handleTap">回到指定位置</button><view class "ControlHeight…

js 小数相乘后,精度缺失问题,记录四舍五入,向下取整

在做项目的时候&#xff0c;有一个计算金额的&#xff0c;结果发现计算的金额总是缺失0.01&#xff0c;发现相乘的时候&#xff0c;会失去精度&#xff0c;如图所示。被这整的吐血&#xff0c;由于计算逻辑由前端计算&#xff0c;所以传值后端总出错(尽量后端计算)。 还发现to…

9月12日作业

作业代码 #include <iostream>using namespace std;class Shape { protected:double cir;double area; public://无参构造Shape() {cout<<"无参构造"<<endl;}//有参构造Shape(double c, double a):cir(c), area(a){cout<<"有参构造&quo…

IDEFICS 简介: 最先进视觉语言模型的开源复现

我们很高兴发布 IDEFICS ( Image-aware Decoder Enhanced la Flamingo with Ininterleaved Cross-attention S ) 这一开放视觉语言模型。IDEFICS 基于 Flamingo&#xff0c;Flamingo 作为最先进的视觉语言模型&#xff0c;最初由 DeepMind 开发&#xff0c;但目前尚未公开发布…

极简B站直播录制工具 录播姬 2.9.0,支持自动批量录制、弹幕录制等

录播姬 是一个简单好用免费开源的直播录制工具&#xff0c;支持自动批量录制、弹幕录制、实时监控直播间状态&#xff0c;直接获取直播流&#xff0c;非录制屏幕&#xff0c;没有二次压制 软件特点 使用简单&#xff1a;粘贴房间号或房间链接即可开录 自动录制&#xff1a;主…

LeetCode 28. 找出字符串中第一个匹配项的下标

文章目录 一、题目二、C# 题解 一、题目 给你两个字符串 haystack 和 needle &#xff0c;请你在 haystack 字符串中找出 needle 字符串的第一个匹配项的下标&#xff08;下标从 0 开始&#xff09;。如果 needle 不是 haystack 的一部分&#xff0c;则返回 -1 。 点击此处跳转…

Java 内存模型(JMM)

1.概述 JMM&#xff0c;全称 Java Memory Model&#xff0c;中文释义Java内存模型 对于 Java 程序员来说&#xff0c;在虚拟机自动内存管理机制下&#xff0c;不再需要像C/C程序开发程序员那样为每一个 new 操作去写对应的 delete/free操作&#xff0c;不容易出现内存泄漏和内…

LeetCode 2596. 检查骑士巡视方案【数组,模拟】1448

本文属于「征服LeetCode」系列文章之一&#xff0c;这一系列正式开始于2021/08/12。由于LeetCode上部分题目有锁&#xff0c;本系列将至少持续到刷完所有无锁题之日为止&#xff1b;由于LeetCode还在不断地创建新题&#xff0c;本系列的终止日期可能是永远。在这一系列刷题文章…

LeetCode 53. 最大子数组和

题目链接 力扣&#xff08;LeetCode&#xff09;官网 - 全球极客挚爱的技术成长平台 题目解析 使用动态规划的方法进行解决&#xff0c;我们创建一个dp表&#xff0c;用来记录以该下标为结尾的最大子数组。然后每次存dp表的时候&#xff0c;进行取最大值。最终返回最大值。 由…

二叉排序树(BST)的算法分析以及基本操作(结点的查询,插入,删除)

1.二叉排序树的定义 二叉排序树&#xff0c;又称二叉查找树&#xff08;BST&#xff0c;Binary Search Tree) 默认不允许两个结点的关键字相同。 1.二叉排序树的性质: 任意一棵二叉排序树的子树的结点大小都满足“左小右大”。 左子树上所有结点的关键字均小于根结点的关键…

基于Python和mysql开发的智慧校园答题考试系统(源码+数据库+程序配置说明书+程序使用说明书)

一、项目简介 本项目是一套基于Python和mysql开发的智慧校园答题考试系统&#xff0c;主要针对计算机相关专业的正在做毕设的学生与需要项目实战练习的Python学习者。 包含&#xff1a;项目源码、项目文档、数据库脚本等&#xff0c;该项目附带全部源码可作为毕设使用。 项目都…