【C++】vector的简单模拟实现

news2024/9/21 21:12:42

目录

一、vector的基本实现机制:

二、vector的部分接口模拟实现:

1、构造与析构:

1、普通构造:

2、拷贝构造:

3、析构函数:

2、关于扩容:

1、reserve:

2、resize

3、增删查改:

1、在pos位置插入:

2、[]符号访问修改:

3、删除pos位置的值:

4、重载=运算符:


一、vector的基本实现机制:

如上所示,其主要由三个成员变量:start,finish,endofstoratge,这三个组成,分别是指向头,有效数据的尾,总容量的尾,

start和finish之间的数据就是size,

start和endofstorage之间的是capacity。

所以,可以用两个函数来直接找到size与capacity,最好也用const修饰,这样的话就不会有权限放大的问题的。

底层其实就是顺序表,也可以理解为数组。

在模拟实现中,要尽量保持和数据库一样的,所以部分使用迭代器的方法来实现,在自己模拟实现的时候推荐在自己定义的命名空间里面进行,这样的话,可以和标准库里面的分开。

在实现迭代器的时候,要实现两个版本的迭代器,一个是正常版本,另外一个是const版本,这样的话就不会有权限放大的问题了。

namespace ppr
{
	template<class T>
	class vector
	{
	public:
		typedef T* iterator;
		typedef const T* const_iterator;
		
		iterator begin() 
		{
			return _start;
		}

		iterator end()
		{
			return _finish;
		}

		const_iterator begin() const
		{
			return _start;
		}

		const_iterator end() const
		{
			return _finish;
		}

        size_t size() const
        {
        	return _finish - _start;
        }

        size_t capacity() const
        {
	        return _endofstorage - _start;
        }
	private:
		iterator _start;
		iterator _finish;
		iterator _endofstorage;
	};

二、vector的部分接口模拟实现:

1、构造与析构:

1、普通构造:

在构造的时候初始化即可:

vector()
	:_start(nullptr)
	, _finish(nullptr)
	, _endofstorage(nullptr)
{}

2、拷贝构造:

思路:

通过capacity()的大小来开辟一个新空间

然后将目标所指向的空间拷贝过去,拷贝目标size的大小,

最后将finish和endofstorage修改为目标大小

vector(const vector<T>& v)
	:_start(nullptr)
	, _finish(nullptr)
	, _endofstorage(nullptr)
{
	_start = new T[v.capacity()];
	memcpy(_start, v._start.sizeof(T) * v.size());
	_finish = _start + v.size();
	_endofstorage = _start + v.capacity();
}

3、析构函数:

思路:

首先检查_start指向的空间是否为空,如果不是空就可以析构,

析构的话,直接delete[] _start即可,

最后将那三者都置为空

~vector()
{
	if (_start != nullptr)
	{
		delete[] _start;
		_start = _finish = _endofstorage = nullptr;
	}
}

2、关于扩容:

1、reserve:

思路:

是采用深拷贝来进行扩容的,所以就要新指针来指向新空间。

首先,根据传过来的n来决定要开辟多少大小的空间就new多大的空间,给一个指针指向。

然后将原来的空间拷贝size过去,释放原来的空间。

最后将原来的start指向新空间,finish和endofstorage搞为新的大小。

注意:

必须在最开始之前将原来的size存好,不然在后面计算finish会出问题。

void reserve(size_t n)
{
	size_t sz = size();
	if (n > capacity())
	{
		T* tmp = new T[n];
		if (_start != nullptr)
		{
			memcpy(tmp, _start, sizeof(T) * sz);
			delete[] _start;
		}
		_start = tmp;
		_finish = _start + sz;
		_endofstorage = _start + n;
	}
}

2、resize

思路:

首先了解两点:
1、n小于当前的大小,那么则保留前n个元素,去除超过的部分。

2、n大于当前的大小,那么在结尾插入适合数量的元素使得整个容器大小达到n。如果给出val,插入的新元素全为val,否则,执行默认构造函数。

那么就可有:

首先进行判断:如果n小于此时的size(),那么直接调整finish即可,

否则,用reserve进行扩容,在用while循环进行赋值。

注意:

这里给val的缺省参数不能够是0,因为我不知道这个val是什么类型,(如果是0就不能够初始化string,vector等等)所以要用T(),这样就会调用它的默认构造,如果是内置类型,那么在有模版后就会对内置类型进行升级,支持内置类型有这种构造。

void resize(size_t n,const T& val = T())
{
	if (n < size())
	{
		_finish = _start + n;
	}
	else
	{
		reserve(n);

		while (_finish != _start + n)
		{
			*_finish = val;
			_finish++;
		}
	}
}

3、增删查改:

1、在pos位置插入:

思路:

首先断言检查:pos位置必须在start和finish中间,

接着写扩容逻辑,然后通过while循环来挪动pos位置之后的数据,

最后将pos位置修改为所插入的数据,修改finish即可。

iterator insert(iterator pos ,const T& x)
{
	assert(pos >= _start && pos <= _finish);

	if (_finish == _endofstorage)
	{
		size_t len = pos - _start;

		size_t newcapacity = capacity() == 0 ? 4 : capacity() * 2; 
		reserve(newcapacity); 

		pos = _start + len;
	}

	iterator end = _finish - 1;
	while (end >= pos)
	{
		*(end + 1) = *end;
		--end;
	}

	*pos = x;
	_finish++; 

	return pos;
}

2、[]符号访问修改:

断言要保证pos不能大于size,直接返回pos所在的位置即可。

T& operator[](size_t pos)
{
	assert(pos<size());

	return _start[pos];
}

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

	return _start[pos];
}

3、删除pos位置的值:

思路:

首先断言,要保证在_start和在_finish之间,接着定义一个pos+1位置的迭代器,将后面的都向前挪动一个位置,最后改变finish即可完成。

为了和库里面保持一致和平台的可移植性,我们需要给一个返回值,返回删除位置的下一个位置(就是pos的位置)。

iterator erase(iterator pos)
{
	assert(pos >= _start && pos < _finish);

	iterator it = pos + 1;
	while (it != _finish)
	{
		*(it - 1) = *it;
		++it;
	}
	--_finish;
	return pos;
}

4、重载=运算符:

思路:
先通过形参的拷贝,接着交换this和v的空间,最后返回this即可。

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

vector<T>& operator=(vector<T> v)
{
	swap(v);
	return *this;
}

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

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

相关文章

SpringCloud开发实战(六):Feign的最佳实践

目录 SpringCloud开发实战&#xff08;一&#xff09;&#xff1a;搭建SpringCloud框架 SpringCloud开发实战&#xff08;二&#xff09;&#xff1a;通过RestTemplate实现远程调用 SpringCloud开发实战&#xff08;三&#xff09;&#xff1a;集成Eureka注册中心 SpringCloud开…

基于SpringBoot的高校BBS在线互动论坛系统

&#x1f4a5;&#x1f4a5;源码和论文下载&#x1f4a5;&#x1f4a5;&#xff1a;基于SpringBoot的高校BBS在线互动论坛系统-源码论文报告数据库.rar 1. 系统介绍 本论文设计并实现了一个基于Spring Boot和Vue的校园论坛系统&#xff0c;该系统分为用户和管理员两个角色。用户…

9/4 链表-力扣 234、19

234.回文链表 给你一个单链表的头节点 head &#xff0c;请你判断该链表是否为回文链表&#xff1b;如果是&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 输入&#xff1a;head [1,2,2,1] 输出&#xff1a;true 思考&#xff1a;链表遍历只能从前往后&a…

Android studio 更换下载的gradle

首先我们下载gradle 打个比方如果我们下载了一个github上的项目&#xff0c;而它使用的是gradle-6.5-bin.zip https://services.gradle.org/distributions/gradle-6.5-bin.zip 用浏览器去下载&#xff0c;可能需要翻墙 解压到电脑里 找到setting里的这一项&#xff0c;设置…

plc1200 weiluntong001

快接口 快代码 main代码 电脑IP地址 编译&#xff0c;启动仿真&#xff0c;下载到仿真PLCsim 必要时候可以设备离线。 打开并监视块。 打开netto plcsim 添加 本机IP&#xff0c;选择&#xff0c;双击。 PLC启动仿真之后&#xff0c;出现这个IP地址&#xff0…

88、k8s之pv+pvc

一、pv和pvc pv pv&#xff1a;Persistent volume 是k8s虚拟化的存储资源&#xff0c;实际上就是存储&#xff0c;例如本地的硬盘&#xff0c;网络文件系统&#xff08;nfs&#xff09; lvm RAID oss&#xff08;ceph&#xff09; 云存储。 pvc pvc&#xff1a;Persisten…

关于SPI通信失败的一种情况(CRC校验不匹配的问题)

问题 该项目中&#xff0c;使用外置的ADC芯片采集电压电流&#xff0c;主控MCU通过SPI与ADC芯片通信。调试时&#xff0c;SPI通信一直失败&#xff0c;与之前成功的项目对比&#xff0c;发现是SPI配置的问题。 void MX_SPI2_Init(void) {/* USER CODE BEGIN SPI2_Init 0 *//*…

2024.9计算机视觉设计开发工程师专项培训通知

为进一步贯彻落实中共中央印发《关于深化人才发展体制机制改革的意见》和国务院印发《关于“十四五”数字经济发展规划》等有关工作的部署要求&#xff0c;深入实施人才强国战略和创新驱动发展战略&#xff0c;加强全国数字化人才队伍建设&#xff0c;持续推进人工智能从业人员…

fastadmin 文件上传腾讯云

1-安装腾讯云SDK composer require qcloud/cos-sdk-v5 2-腾讯云配置 <?phpnamespace app\common\controller;use Qcloud\Cos\Client; use think\Controller; use think\Db;class Tencent extends Controller {/*** 上传文件* param $config* param $key* return array*/p…

微信公众号《GIS 数据工程:开始您的 ETL 之旅 》 文章删除及原因

微信公众号多次限制付费文章发布&#xff0c;不太明确其原因。我猜可能是得罪了某位大神&#xff0c;这倒是也不是不可能。我这说话口无遮拦&#xff0c;得罪几个人偶尔搞我一下也是应该的 。当然也可能是部分喜欢白嫖的网友一看我收费就不太高兴&#xff0c;偶尔做点小动作也是…

Windows系统下苹果虚拟机系统的安装

前言 搞苹果软件开发&#xff0c;未必要购买贵的苹果电脑。可以安装黑苹果系统&#xff0c;也可以安装VMware的苹果虚拟机。而且通过我的实践发现&#xff0c;目前苹果虚拟机的效果很不错。 1、参考文档链接 VM虚拟机怎么安装mac os&#xff1f;&#xff08;全教程&#xff0…

【LeetCode】03.无重复字符的最长子串

题目要求 做题链接3.无重复字符的最长子串 解题思路 我们通过参考给出的输入很容易就会从每一个字符开始&#xff0c;看看最长能延续多长。我们通过画图发现一旦一个字符可以延续到另一个字符&#xff0c;那么我们就不需要考虑他中间仍然存在字符重复的问题。因此而后我们发…

部署mongosh教程

1、上传软件包 将软件包上传到/usr/local目录下 部署 2.1 解压 tar zxvf mongosh-2.3.0-linux-x64.tgz 2.2 修改名称 mv mongosh-2.3.0-linux-x64/ mongosh 2.3 将 bin 目录中 mongosh 二进制文件复制到 PATH 变量中列出的目录中 sudo cp mongosh /usr/local/bin/ sudo cp …

第九届“创客中国”生成式人工智能中小企业创新创业大赛招商推介圆满落幕

金秋九月,丹桂飘香。9月2日晚,第九届“创客中国”生成式人工智能(AIGC)中小企业创新创业大赛招商推介会在南昌高新区艾溪湖畔成功举办。南昌市政府副秘书长、办公室党组成员陈吉炜出席并致辞。市中小企业局党组书记、市工信局党组书记、局长骆军出席。南昌高新区党工委委员、管…

16 C语言连接

使用c语言连接mysql&#xff0c;需要使用mysql官网提供的库&#xff0c;可以在官网下载 准备工作&#xff1a; 保证mysql服务有效 官网下载合适的mysql connect库 也可以直接安装mysql服务 yum install -y mysql-devel Connector/C使用 库格式如下&#xff1a; [hbMiWiFi-R1…

无线麦克风推荐,无线麦克风十大名牌,领夹麦克风十大品牌推荐

在音频创作的专业舞台上&#xff0c;无线领夹麦克风不仅是声音捕捉的利器&#xff0c;更是创作者表达情感的桥梁。然而&#xff0c;市场上琳琅满目的产品让人眼花缭乱&#xff0c;不少劣质麦克风不仅收音效果大打折扣&#xff0c;还常因信号不稳、噪音干扰而破坏了作品的纯净度…

使用NI CANopen与多个节点进行通信

要使用N CANopen同时与多个节点进行通信&#xff0c;可以利用CANopen Library Toolkit提供的不同VI。以下是详细的步骤说明&#xff1a; 了解VI的功能&#xff1a; NMT Write.vi&#xff1a; 该VI用于向特定节点或所有节点发送网络管理&#xff08;NMT&#xff09;命令。使用No…

Linux之多线程概念

目录 线程概念 线程共享的资源 线程独有的资源 线程优点 线程缺点 在之前&#xff0c;我们已经学习了进程相关的知识点&#xff0c;如进程的基本概念和基本操作&#xff0c;本期我们将开始进行线程的学习&#xff0c;探索线程和进程的关系。 线程概念 在学习线程之前…

镭速助力构筑ICT安全高效的大文件传输

在数字化浪潮的推动下&#xff0c;信息通信技术&#xff08;ICT&#xff09;行业正以惊人的速度发展&#xff0c;随之而来的是跨国、跨区域文件传输需求的激增。然而&#xff0c;传统的大文件传输方法已难以满足当前ICT行业的需求&#xff0c;面临着诸多挑战&#xff0c;如跨国…

多用户B2B2C商城源码+短视频直播+APP+小程序+H5

店铺管理 店铺列表&#xff0c;新店铺审核&#xff0c;地址管理&#xff0c;服务管理&#xff0c;运费模版&#xff0c;品牌管理 订单监管 视频订单&#xff0c;拼团订单&#xff0c;评论管理&#xff0c;退款管理 装修商城 模版管理&#xff0c;页面管理&#xff0c;全局…