C++:模拟实现vector

news2024/9/29 22:43:18

目录

成员变量与迭代器

size

capacity

empty

迭代器有关函数

实现默认成员函数的前置准备

reserve

​编辑

​编辑

push_back

构造函数

无参构造

迭代器区间构造

n个val来进行构造 

析构函数

拷贝构造函数

赋值重载

增删查改

clear

resize

pop_back

insert

erase

重载[]


成员变量与迭代器

我们还是需要在一个命名空间里模拟实现vector,防止和标准库里的起冲突。

namespace zh
{
	template<class T>
	class vector
	{
	public:
		typedef T* iterator;
		typedef const T* const_iterator;

    private:
	    iterator _start = nullptr;
	    iterator _finish = nullptr;
	    iterator _end_of_storage = nullptr;
    };
}

解释说明:

1.vector是一个非常通用的容器,是一个动态大小的数组,可以存储任意类型的元素,并且能够自动调整大小以适应元素的添加和删除。所以我们的模拟实现要写成类模板

2.vector可以看做顺序表的升级,但是模拟实现vector跟我们以往实现顺序表有所不同,顺序表是使用一个动态开辟的数组、数组有效元素个数size和数组容纳最大有效数据的个数capacity维护的,而模拟实现vector需要三个(模板参数)T* 类型的指针,而vector的迭代器功能恰恰又和T*类型指针类似,所以干脆把T*封装成迭代器。当然迭代器需要有两个版本,普通版本和const版本。

3.参数的含义

_start指向数组首元素,_finish指向最后一个有效元素的下一个位置, _end_of_storage指向数组空间末尾。

通过三个指针也可以模拟出size和capacity的功能。

size

返回有效数据个数的函数

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

capacity

返回数组最大容纳有效数据个数(容量大小)的函数。

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

empty

判断数组是否为空,判断_start与_finish是否相等即可

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

迭代器有关函数

主要实现begin函数和end函数

iterator begin()
{
	return _start;
}

iterator end()
{
	return _finish;
}

const_iterator begin() const
{
	return _start;
}

const_iterator end() const
{
	return _finish;
}

实现默认成员函数的前置准备

reserve

用于vector数组空间不足时扩容的函数(扩容成n个空间)。

void reserve(size_t n)
{
	if (n > capacity())                        //n大于数组容量才扩容
	{
		size_t oldsize = size();               //用oldsize避免新_start和老_finish的问题
                                                
		T* tmp = new T[n];
		//memcpy(tmp, _start, size() * sizeof(T));  //这里是浅拷贝,如果是内置类型,没问题 
                                                    //如果vector存的是自定义类型,就是大坑
		for (size_t i = 0; i < oldsize; ++i)
		{
			tmp[i] = _start[i];
		}
		delete _start;                 //这里delete_start,_finish 和_end_of_storage是野指针

		//更新成员变量
		_start = tmp;
		_finish = tmp + oldsize;               
		_end_of_storage = tmp + n;
	}
}

reserve有几个问题需要注意:

1.开空间的时候要使用new而不要用malloc,因为malloc只是去开空间,不会去调用构造函数。

2.新_start和_finish的问题。

错误示范。

将原有数据拷贝到新空间后,释放了旧空间的资源,_strat指向了新的空间,但是_finish和_end_of_storage还是指向旧空间,这两个指针就变成野指针了。而最关键的是_finish不能被正确赋值。

3.memcpy浅拷贝问题

memcpy(tmp, _start, size() * sizeof(T));

memcpy是浅拷贝,如果vector存的是内置类型,那么浅拷贝就没有问题,如果存的是自定义类型,那浅拷贝就是个大坑。假如vector存的是string类型,那么扩容时,将数据从旧空间拷贝到新空间时,因为是浅拷贝,所以两个空间里的string的_str是同一个地址释放旧空间的时候就连带这把新空间的资源也释放了

这样就扩容失败了,因为你把原空间的数据丢失了,而且搞不好有可能程序还会崩溃。

要解决这个问题,我们就得手动实现深拷贝, 因为new出来的空间如果是自定义类型的话就自动调用构造函数初始化了,所以这里走的是赋值重载来实现深拷贝

push_back

用于在数组末尾尾插一个元素的函数。

void push_back(const T& x)
{
	//插入之前先判断空间是否足够
    if (_finish == _end_of_storage)
	{
		reserve(capacity() == 0 ? 4 : 2 * capacity());
	}

    //插入元素,更新_finish
	*_finish = x;
	_finish++;
}

构造函数

vector的构造函数我们实现无参构造迭代器区间构造n个val构造

无参构造

无参构造其实我们并不需要写,因为已经在成员变声明时给了缺省值,编译器自动生成的无参构造函数走初始化列表满足需求了。但是由于我们写了其他构造函数编译器就不自动生成了

这里时候可以自己写无参构造,也可以用default强制编译器生成(C++11的用法)。

//构造
/*vector()
{}*/

//c++11 强制生成构造
vector() = default;

迭代器区间构造

//类模板的成员函数,还可以继续是函数模版
template<class InputIerator>
vector(InputIerator first, InputIerator last)
{
	while (first != last)
	{
		push_back(*first);	
		++first;
	}
}

这里给这个函数再套一层模板是为了让vector不仅能用vector的迭代器区间构造,还能用其他容器(list、string等)的迭代器来进行构造

这里又有个问题,就是while循环判断条件的!=不能改成<,因为<对于vector的迭代器时可以的,但是对于其他容器的迭代器,如list,last不一定比first要大

n个val来进行构造 

vector(size_t n, const T& val = T())
{
	//先开好空间
	reserve(n);
	for (size_t i = 0; i < n; ++i)
	{
		push_back(val);
	}
}

使用的时候val可能不传参,所以要给缺省值。

因为val的类型不确定,可能是内置类型,也可能是自定义类型。

在不传参使用缺省值时

对于自定义类型,比如strng,先调用构造函数构造一个匿名对象,再拷贝构造给val。(编译器会优化,直接对val进行构造),这样val就有了缺省值

对于内置类型,本来是没有构造函数的说法的,但是为了适应这里,也支持类似类那种使用构造函数初始化的方式。

int a = int();
int b = int(2);
int c(3);
cout << a << endl;
cout << b << endl;
cout << c << endl;

析构函数

直接delete就可以了,把三个迭代器置空。

//析构
~vector()
{
	if (_start)
	{
		delete[] _start;
		_start = _finish = _end_of_storage = nullptr;
	}
}

拷贝构造函数

先开好空间,然后尾插就可以了。

//拷贝构造
vector(const vector<T>& v)
{
	reserve(v.size());
	for (auto& e : v)
	{
		push_back(e);
	}
}

赋值重载

首先实现一个交换函数,然后传值调用,将两个对象交换即可。

//void swap(vector& v) 可以这样写
void swap(vector<T>& v) 
{
	std::swap(_start, v._start);
	std::swap(_finish, v._finish);
	std::swap(_end_of_storage, v._end_of_storage);
}

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

增删查改

clear

不需要真的删除,直接将更改_finish的值即可。

void clear()
{
	_finish = _start;
}

resize

控制有效数据个数。

  1. 若n < size,直接将_finish更改为_start + n即可。
  2. 若_size < n < capacity或者n > capacity,直接扩容成n个空间(空间足够就不会扩容),从_finish拷贝足够数量的val即可。
void resize(size_t n, T val = T())
{
	if (n < size())
	{
		_finish = _start + n;
	}
	else
	{
		reserve(n);
		while (_finish != _start + n)           
		{
			*_finish = val;
			++_finish;
		}
	}
}

pop_back

先判断数组是否为空,尾删一个元素,_finish-- 即可。

void pop_back()
{
	//判断下数组是否为空
	assert(!empty());
	--_finish;
}

insert

在pos位置插入一个元素。

iterator insert(iterator pos, const T& x) //pos不会为0,因为是有效的迭代器
{
	assert(pos >= _start);
	assert(pos <= _finish);

	if (_finish == _end_of_storage)                   //涉及到扩容,pos会失效,pos指向原来的空间
	{
		size_t len = pos - _start;
		reserve(capacity() == 0 ? 4 : 2 * capacity());
		pos = _start + len;
	}
	iterator end = _finish - 1;
	while (end >= pos)
	{
		*(end + 1) = *end;
		--end;
	}

	//插入元素,更新
	*pos = x;
	++_finish;
	return pos;
}

注意的问题:

1.如果插入涉及到了扩容,要提前把pos相对于首元素的相对长度记录下来,扩容完毕后更新pos。因为扩容会导致pos失效。

2.插入之后要返回新元素的迭代器。(这里其实也算迭代器是失效了,因为pos指向的元素发生了更改,迭代器失效了就不要在使用了。)

erase

删除pos位置的元素,删除完后返回删除元素下一位置的迭代器

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

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

	return pos;
}

抛出一个问题,利用迭代器删除vector中所有的偶数。

错误做法

auto it = v.begin();
while (it != v.end())
{
	if (*it % 2 == 0)
	{
		it = v.erase(it);
	}
    
	it++;		
}

删完一个偶数后,it已经是下一元素的迭代器了,it不需要++了。

正确做法

auto it = v.begin();
while (it != v.end())
{
	if (*it % 2 == 0)
	{
		it = v.erase(it);
	}
	else
	{
		++it;
	}
}

重载[]

为了方便访问和修改数组中的元素。

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

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

通用打印容器函数,套一层模板即可。

注意:

​
template<class Container>
void print_Container(const Container& v)
{
	//typename vector<T>::const_iterator it = v.begin();   //typename标定为类型                          
    //从没有实例化的类模板取出来的可能是类型或者成员变量,编译器无法区分
	auto it = v.begin();                       

	while (it != v.end())
	{
		cout << *it << ' ';
		++it;
	}
	cout << endl;

	/*for (auto num : v)
	{
		cout << num << ' ';
	}
	cout << endl;*/
}

​

未实例化的类取出来的有可能是类型或者成员变量,要加关键字typename告诉编译器是类型不加的话会发生编译错误

当然直接用auto更方便。


拜拜,下期再见😏

摸鱼ing😴✨🎞

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

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

相关文章

Go语言匿名字段使用与注意事项

1. 定义 Go语言支持一种特殊的字段只需要提供类型而不需要写字段名的字段&#xff0c;称之为匿名字段或者嵌套字段。 所谓匿名字段实际上是一种结构体嵌套的方式&#xff0c;所以也可以称作嵌套字段。 这种方式可以实现组合复用&#xff0c;即通过匿名字段&#xff0c;结构体…

开放原子开源基金会网站上的开源项目Opns存在缓冲区溢出缺陷

最近在开放原子开源基金会网站上&#xff0c;看到一些开源项目&#xff0c;之前分析出华为的鸿蒙操作系统代码&#xff0c;没有发现有价值的安全漏洞。现在&#xff0c;下载上面的Onps开源网络协议栈&#xff0c;既然是通讯所使用的软件&#xff0c;其质量应该值得信任呢&#…

LeetCode[中等] 238. 除自身以外数组的乘积

给你一个整数数组 nums&#xff0c;返回 数组 answer &#xff0c;其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法&#xff0c;且在 O(n) 时间复杂…

828华为云征文 | 云服务器Flexus X实例:向量数据库 pgvector 部署,实现向量检索

目录 一、什么是向量数据库 pgvector &#xff1f; 二、pgvector 部署 2.1 安装 Docker 2.2 拉取镜像 2.3 添加规则 三、pgvector 运行 3.1 运行 pgvector 3.2 连接 pgvector 3.3 pgvector 常见操作 四、总结 本篇文章通过 云服务器Flexus X实例 部署向量数据库 pgve…

什么情况?上交所服务器被你们给买崩了?

号主&#xff1a;老杨丨11年资深网络工程师&#xff0c;更多网工提升干货&#xff0c;请关注公众号&#xff1a;网络工程师俱乐部 上午好&#xff0c;我的网工朋友。 9月27日早上&#xff0c;A股市场迎来了一波前所未有的火爆行情&#xff0c;成交量激增&#xff0c;市场情绪高…

51单片机的光照强度检测【proteus仿真+程序+报告+原理图+演示视频】

1、主要功能 该系统由AT89C51/STC89C52单片机LCD1602显示模块光照传感器按键蜂鸣器LED等模块构成。适用于光照强度检测、光照强度测量报警等相似项目。 可实现功能: 1、LCD1602实时显示光照强度信息 2、光照强度传感器&#xff08;电位器模拟&#xff09;采集光照信息 3、可…

鸿蒙开发(NEXT/API 12)【硬件(振动开发1)】振动

通过最大化开放马达器件能力&#xff0c;振动器模块服务拓展了原生马达服务&#xff0c;实现了振动与交互融合设计&#xff0c;从而打造出细腻精致的一体化振动体验和差异化体验&#xff0c;提升用户交互效率、易用性以及用户体验&#xff0c;并增强品牌竞争力。 运作机制 Vi…

小程序原生-地理位置授权用户拒绝后的解决方案

在开发的过程中&#xff0c;我们会发现一个问题&#xff1a; 在调用 wx.getLocation() 获取用地理位置时&#xff0c;如果用户选择拒绝授权&#xff0c;代码会直接抛出错误。在拒绝授权以后&#xff0c;再次调用 wx.getLocation() 时&#xff0c;就不会在弹窗询问用户是否允许…

【国庆要来了】基于Leaflet的旅游路线WebGIS可视化实践

前言 转眼2024年的国庆节马上就要来临了&#xff0c;估计很多小伙伴都计划好了旅游路线。金秋十月&#xff0c;不管是选择出门去看看风景&#xff0c;还是选择在家里看人。从自己生活惯了的城市去别人生活惯了的城市&#xff0c;去感受城市烟火、去感受人文风景&#xff0c;为2…

Windows环境 源码编译 FFmpeg

记录一下windows环境纯代码编译ffmeg的过程&#xff01; 目录 一、安装MSYS2 1.下载安装 2.配置 3.修改源 4.测试与更新 二、安装其他必要工具 1.安装MinGW-w64 2.安装git 3..安装make等工具 4.编译前的其他准备工作 ①.重命名link.exe ②.下载和安装YASM ③.安装…

ADC的原理

一、介绍 模数转换&#xff0c;即Analog-to-Digital Converter&#xff0c;常称ADC&#xff0c;是指将连续变量的模拟信号转换为离散的数字信号的器件&#xff0c;比如将模温度感器产生的电信号转为控制芯片能处理的数字信号0101&#xff0c;这样ADC就建立了模拟世界的传感器和…

tomcat服务器控制台乱码(超简单,其中一种方式)

一、问题 为解决前&#xff1a; 是这样子&#xff0c;控制台中文乱码 解决后&#xff1a; 二、解决 在tomcat的安装目录下&#xff0c;找到conf ---》logging.properties 点击去进行编辑 重点&#xff01;&#xff01;&#xff01; 编辑完之后&#xff0c;保存 如何返回到…

文献阅读9.29

目录 基于物理的神经网络在河流淤积模拟中的应用 文献摘要 讨论|结论 理论介绍 PINN 实验方程 Ansys中的数学模型 实验设置 基于物理的神经网络在河流淤积模拟中的应用 文献摘要 本文从水动力学和污染物运移动力学的基本原理出发&#xff0c;全面探讨了用于研究泥沙淤…

六、动画系统

一、动画事件 在此处实现&#xff1a;动画播放至此处&#xff0c;人物角色移动 1、添加动画事件 选择对应的动画 右键动画部分&#xff0c;选择添加动画事件 在人物身上挂载了对应的动画系统&#xff0c;需要在代码中&#xff0c;设置一个函数&#xff0c;作为动画事件 执行…

【ADC】使用仪表放大器驱动 SAR 型 ADC 时的输入输出范围

概述 本文学习于TI 高精度实验室课程&#xff0c;介绍使用仪表放大器时 SAR ADC 驱动放大器的注意事项。具体包括&#xff1a;介绍如何使用仪表放大器设计数据转换器驱动电路。 仪表放大器&#xff08;Instrumentation Amplifier&#xff0c;下文简称 INA&#xff09;可抑制输…

自定义knife4j访问路径

文章目录 本文档只是为了留档方便以后工作运维&#xff0c;或者给同事分享文档内容比较简陋命令也不是特别全&#xff0c;不适合小白观看&#xff0c;如有不懂可以私信&#xff0c;上班期间都是在得 原由&#xff0c;嫌弃doc.html 太大众 直接重定向&#xff0c;直接上代码了 p…

【Linux】Linux内核结构基础

Linux内核结构基础 一、Linux内核结构介绍二、Linux内核结构☆ Linux内核结构框图Linux系统架构 三、认识驱动为什么学驱动文件名与设备号open函数过程实例详解 四、shellshell脚本 参考博文&#xff1a; 【Linux】内核结构&#xff08;全&#xff09;linux内核结构介绍&#…

理解Python闭包概念

闭包并不只是一个python中的概念&#xff0c;在函数式编程语言中应用较为广泛。理解python中的闭包一方面是能够正确的使用闭包&#xff0c;另一方面可以好好体会和思考闭包的设计思想。 1.概念介绍 首先看一下维基上对闭包的解释&#xff1a; 在计算机科学中&#xff0c;闭包…

【动态规划-分组背包】【hard】力扣2218. 从栈中取出 K 个硬币的最大面值和

一张桌子上总共有 n 个硬币 栈 。每个栈有 正整数 个带面值的硬币。 每一次操作中&#xff0c;你可以从任意一个栈的 顶部 取出 1 个硬币&#xff0c;从栈中移除它&#xff0c;并放入你的钱包里。 给你一个列表 piles &#xff0c;其中 piles[i] 是一个整数数组&#xff0c;分…

OpenEuler虚拟机安装保姆级教程 | 附可视化界面

0x00 系统介绍 在 2019 年 7 月 19 日&#xff0c;华为宣布要在年底正式开源 openEuler 操作系统&#xff1b;在半年后的 12 月 31 日&#xff0c;华为正式开源了 openEuler 操作系统&#xff0c;邀请社区开发者共同来贡献。 一年后&#xff0c;截止到 2020 年12 月 25日&…