C++vector模拟实现

news2024/11/23 18:53:21

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,capacity
			int 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,capacity
			int 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,capacity
			int 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,capacity
			int 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);
			//更新pos
			pos = _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.coloradmin.cn/o/1006020.html

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

相关文章

SpringBoot整合Redis,基于Jedis实现redis各种操作

前言&#xff08;三步教你学会redis&#xff0c;主打一个实用&#xff09; springboot整合redis步骤&#xff0c;并基于jedis对redis数据库进行相关操作&#xff0c;最后分享非常好用、功能非常全的redis工具类。 第一步&#xff1a;导入maven依赖 <!-- springboot整合re…

C++之智能指针shared_ptr死锁问题(二百)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

Hadoop生态圈中的Flume数据日志采集工具

Hadoop生态圈中的Flume数据日志采集工具 一、数据采集的问题二、数据采集一般使用的技术三、扩展&#xff1a;通过爬虫技术采集第三方网站数据四、Flume日志采集工具概述五、Flume采集数据的时候&#xff0c;核心是编写Flume的采集脚本xxx.conf六、Flume案例实操1、采集一个网络…

卫星地图-航拍影像-叠加配准套合(ArcGIS版)

卫星地图-航拍影像-叠加配准套合(ArcGIS版) 发布时间&#xff1a;2018-01-17 版权&#xff1a;BIGEMAP 第一步 工具准备 BIGEMAP地图下载器&#xff1a;Bigemap系列产品-GIS行业基础软件kml\shp 相关教程&#xff1a;CAD文件直接导入BIGEMAP进行套合配准&#xff08;推荐&am…

苹果笔不用原装可以吗?苹果ipad触控笔推荐

很多小伙伴在纠结&#xff0c;是否要购买apple pencil呢&#xff1f;但它的价格太过昂贵&#xff0c;很多学生党都消费不起。答案是不一定的要入手apple pencil的。市面上也是有做得相当不错的平替电容笔。现在无纸化已经快成为我们生活中的一部分&#xff0c;它不仅是可以书写…

新网站怎么优化才能提升排名-彻底解决新网站没收录和排名的问题

新网站怎么优化才能提升排名&#xff1f;辛辛苦苦搭建起网站后&#xff0c;却不知如何优化网站提升排名&#xff0c;也没有一个完整的思路和策略&#xff0c;今天做为行业资深SEO的我&#xff0c;来为大家理清SEO思路&#xff0c;彻底解决新网站没收录和排名的问题&#xff01;…

网站不收录没排名降权怎么处理-紧急措施可恢复网站

网站降权对于SEO人员来说是非常致命的打击&#xff0c;因为网站一旦被搜索引擎降权&#xff0c;排名会严重地下降&#xff0c;网站的流量也会大幅下降&#xff0c;直接影响到收益。而且处理不好的话会导致恢复的时间周期无限拉长&#xff0c;所以网站被降权后我们要第一时间采取…

8月客户文章盘点——累计IF 168.4,平均IF 8.42

客户文章一览 凌恩生物以打造国内一流生物公司为目标&#xff0c;在科研测序领域深耕不辍&#xff0c;吸纳多名在生物信息高级技术人员的加盟&#xff0c;参与并完成多个高科技项目。现已在宏组学、基因组、表观遗传以及蛋白代谢等多组学及联合分析领域积累了深厚经验&#xff…

软件测试总结1

1、 什么是软件测试? 答: 软件测试是在规定的条件下对程序进行操作&#xff0c;以发现错误&#xff0c;对软件质量进行评估。 什么是软件测试&#xff1a; 明确地提出了软件测试以检验是否满足需求为目标。 1、保证软件质量的重要手段 预期 ≈ 实际 2、 软件测试的意义 给…

其它机器访问mysql配置

其它机器访问mysql配置 搜索工具 叫 Everything 一、mysql - 改my.ini 刷脚本 也可能叫其他名字 编辑 bind-address0.0.0.0 编辑然后重启一下mysql服务任务管理器-关掉mysql 编辑 搜索 计算机管理-重启mysql服务 编辑 然后 打开查询&#xff0c;并选择mysql数据&#x…

分享微信聊天记录恢复的3个简单方法!

无论身在何处&#xff0c;微信是我们与家人、好友保持紧密联系的好帮手。微信聊天记录不仅仅只是几句话、几张照片……更多的是聊天记录中承载着的美好回忆。如果不小心将聊天记录删除了怎么办&#xff1f;微信聊天记录恢复的方法有哪些&#xff1f;接下来&#xff0c;小编将以…

解决IntelliJ IDEA执行maven打包,执行java -jar命令提示jar中没有主清单属性

问题场景 IDEA执行mvn clean package -DskipTesttrue命令或者借助工具的Maven菜单进行打包操作&#xff0c;然后执行java -jar app.jar命令后&#xff0c;提示jar中没有主清单属性 D:\WorkSpace\demo\target>java -jar demo-SNAPSHOT.jar demo-SNAPSHOT.jar中没有主清单属性…

flex布局中的几个小技巧

1. flex属性一定要写在父级元素的css属性&#xff0c;比如ul里面&#xff0c;不能写到li上面 ul{display:flex; // flex必须写在父元素ul里面justify-content:space-between;height:75px;li{height:75px;line-height:75px;flex:1;padding-left:23px;a{color:$colorB;fo…

3D数字孪生:从3D数据采集到3D内容分析

数字孪生&#xff08;Digital Twin&#xff09;是物理对象、流程或系统的虚拟复制品&#xff0c;用于监控、分析和优化现实世界的对应物。 这些数字孪生在制造、工程和城市规划等领域变得越来越重要&#xff0c;因为它们使我们能够在现实世界中实施改变之前模拟和测试不同的场景…

Linux操作(查询日志)

目录 前言 查看日志 cat less head tail 小结 前言 之前的linux文章属于入门linux,这篇文章主要是linux在后端开发人员中对日志的的运用.对于linux基础掌握不是很好的小伙伴可以先去看看linux基础操作:Linux系统使用(超详细)_linux操作系统使用_陌上 烟雨齐的博客-CSDN博…

Leetcode152. 连续子数组的最大乘积

力扣&#xff08;LeetCode&#xff09;官网 - 全球极客挚爱的技术成长平台 给你一个整数数组 nums &#xff0c;请你找出数组中乘积最大的非空连续子数组&#xff08;该子数组中至少包含一个数字&#xff09;&#xff0c;并返回该子数组所对应的乘积。 测试用例的答案是一个 32…

Android 系统中适配OAID获取

一、OAID概念 OAID&#xff08;Open Anonymous Identification&#xff09;是一种匿名身份识别标识符&#xff0c; 用于在移动设备上进行广告追踪和个性化广告投放。它是由中国移动通信集 团、中国电信集团和中国联通集团共同推出的一项行业标准 OAID值为一个64位的数字 二、…

【C语言】库宏offsetof

一.offsetof简介 因此,宏offsetof的作用是: 当你传入结构体的类型及其成员时,它会返回该成员在结构体中的偏移量. 二.offsetof的使用 如下,我们使用offsetof打印一下结构体foo中,成员a,成员b及成员c相对于首地址的偏移量分别是多少: #include <stdio.h> #include …

The specified module could not be found.

新电脑运行项目的时候出现了某个包找不到的问题 \\?\D:\guanwnag\cloudquery-website\node_modules\.pnpm\nxnx-win32-x64-msvc16.5.3\node_modules\nx\nx-win32-x64-msvc\nx.win32-x64-msvc.node 引入的路径就感觉有问题 去github上查找原因&#xff0c;发现是没安装 Micro…

代码随想录算法训练营day50|123.买卖股票的最佳时机III|188.买卖股票的最佳时机IV

123.买卖股票的最佳时机III 力扣题目链接 给定一个数组&#xff0c;它的第 i 个元素是一支给定的股票在第 i 天的价格。 设计一个算法来计算你所能获取的最大利润。你最多可以完成 两笔 交易。 注意&#xff1a;你不能同时参与多笔交易&#xff08;你必须在再次购买前出售掉…