[C++]vector类的模拟实现和相关函数的详解

news2025/2/25 22:00:56

文章目录

  • 架构
  • 实现
    • 默认构造函数
      • 构造函数
      • 拷贝构造
        • 为什么不能使用memcpy()进行拷贝(浅拷贝问题)
      • 析构函数
      • 赋值重载
        • =
        • []
    • 迭代器
      • begin && end
    • 操作函数
      • size() && capacity()
      • push_back()
      • reserve()
      • resize()
      • insert()
      • erase()
  • 完整代码

架构

首先由于自定义类不能和已有类重名,所以在自定义命名空间中进行vector类的模拟

namespace aiyimu
{
	template<class T>//类模板
	class vector
	{
	
	public:
		//迭代器
		typedef T* iterator;
		typedef const T* const_iterator;
		//成员函数
		
	private:
		//成员变量
		iterator _start;//起始位置
		iterator _finish;//所含元素末尾
		iterator _endofstorage;//开辟的空间末尾位置
	};
}

实现

默认构造函数

构造函数

无参构造函数

无参构造进行成员函数的初始化即可

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

迭代器构造函数

  1. 定义模板参数:为了使任意类型的迭代器都能使用
  2. 将成员变量初始化
  3. 用while循环遍历迭代器范围[first, last),将范围内的元素逐个插入到vector中,这里使用的push_back()(尾插函数)后面会进行实现
template <class InputIterator>//定义一个模板参数
vector(InputIterator first, InputIterator last)
	:_start(nullptr)
	,_finish(nullptr)
	,_endofstorage(nullptr)
{
	//从first到last构造
	while (first != last)
	{
		push_back(*first);
		++first;
	}
}

拷贝构造

传统写法*

  1. new一个大小为v.capacity()的连续内存空间,将_start指向该空间起始位置,起始位置设置完成,同理更新_finish, _endofstorage (capacity()是返回vector空间大小的函数,后面实现)
vector(const vector<T>& v)
{
	_start = new T[v.capacity()];
	_finish = _start + v.size();
	_endofstorage = _start + capacity();

	//memcpy(_start, v._start, v.size() * sizeof(T));
 	//同reserve()一样,memcpy是浅拷贝,会出现问题
	for (size_t i = 0; i < sz; i++)
	{
		tmp[i] = _start[i];
	}
}

现代写法*

  1. 现代方法 创建了一个临时的vector对象tmp,利用该对象的构造函数和迭代器传入源vector对象v的起始地址和结束地址,构造出一个与v完全相同的新的vector对象tmp。
  2. 同时利用自定义的swap()函数进行拷贝操作(将tmp的元素给*this)
  3. 同样避免了浅拷贝问题
void swap(vector<T>& v)
{
	std::swap(_start, v._start);
	std::swap(_finish, v._finish);
	std::swap(_endofstorage, v._endofstorage);
}

vector(const vector<T>& v)
	:_start(nullptr)
	,_finish(nullptr)
	,_endofstorage(nullptr)
{
	vector<T> tmp(v.begin(), v.end());;//利用构造函数直接将v构造到临时vector中

	swap(tmp);//和this->swap(tmp)相同
}

为什么不能使用memcpy()进行拷贝(浅拷贝问题)

为什么不能使用memcpy()?


如果拷贝的是内置类型的元素,memcpy即高效又不会出错;
但如果拷贝的是自定义类型元素,并且自定义类型元素中涉及到资源管理时,就会出错,因为memcpy的拷贝实际是浅拷贝


具体过程如下图:
当使用memcpy()拷贝后,_start指向v._start所指向的空间,拷贝完后释放原空间,此时_start就变为了野指针。

在这里插入图片描述


析构函数

  1. 如果_start本来为空,不需要进行操作
  2. 不为空进行:释放_start指向的空间,将成员变量指针置空
~vector()
{
	//如果_start不为空,释放空间并置空指针
	if (_start)
	{
		delete[] _start;
		_start = _finish = _endofstorage = nullptr;
	}
}

赋值重载

  1. 赋值重载恰好使用拷贝构造函数中的swap函数直接交换即完成赋值

=

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

[]

  1. 访问下标实现const和非const两种,前者不允许用户通过引用修改vector的内容,后者允许
  2. 首先用assert断言pos不能越界,然后返回下标即可
const T& operator[](size_t pos) const
{
	assert(pos < size());//防止越界
	return _start[pos];
}

T& operator[](size_t pos)
{
	assert(pos < size());//防止越界
	return _start[pos];
}

迭代器

首先我们在类进行了如下操作:分别typedef了两种迭代器(const和非const)

public:
typedef T* iterator;
typedef const T* const_iterator;

begin && end

  1. 使用迭代器时需要使用头尾位置,所以这里定义const/非const的两种函数,直接返回位置即可
const iterator begin() const
{
	return _start;
}

const iterator end() const
{
	return _finish;
}

iterator begin()
{
	return _start;
}

iterator end()
{
	return _finish;
}

操作函数

size() && capacity()

  1. size()capacity()前面已经进行了使用,是为了访问vector的 大小容量
  2. 同样进行const和非const 的实现
//返回此时vector的大小
size_t size()
{
	return _finish - _start;
}
//返回此时vector的容量
size_t capacity()
{
	return _endofstorage - _start;
}

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

push_back()

法一

  1. 首先判断增容和进行增容:用 tmp 临时存储大小为 newCapacity 的内存空间,用for循环完成拷贝操作 (第一次增容不需要拷贝,所以有 if(_start)
  2. 完毕后更新三个指针(_start _finish _endofstorage)
  3. 但更新时需要注意,_start应该在_finish后更改,因为_start的改变会使size()改变,从而使_finish出问题
  4. 在尾部插入x,更新_finish
void push_back(const t& x)
{
	//判断扩容
	if (_finish == _endofstorage)
	{
		size_t newcapacity = capacity() == 0 ? 4 : capacity() * 2;//扩容
		t* tmp = new t[newcapacity];
		
		if (_start)//当_start不为空进来(第一次增容不用进)
		{
			memcpy(tmp, _start, sizeof(t) * size());//将原本存储的数据拷贝到tmp中
			delete[] _start;
		}
		
		_finish = _start + size();
		_start = tmp;//这两行顺序不能变:先赋值_start会改变size()的返回值
		_endofstorage = _start + newcapacity;
	}
	*_finish = x;
	++_finish;
}

法二

  1. 法二在法一的基础上,用sz临时存储size()的值,最后更新_finish直接用sz即可,这样_start的更改顺序不会影响_finish的值
  2. 其余部分和法一思路一致
void push_back(const T& x)
{
	//判断扩容
	if (_finish == _endofstorage)
	{
		size_t newCapacity = capacity() == 0 ? 4 : capacity() * 2;//扩容
		T* tmp = new T[newCapacity];
		size_t sz = size();
		if (_start)//当_start不为空进来(第一次增容不用进)
		{
			memcpy(tmp, _start, sizeof(T) * size());//将原本存储的数据拷贝到tmp中
			delete[] _start;
		}
		_start = tmp;
		_finish = _start + sz;
		_endofstorage = _start + newCapacity;
	}
	*_finish = x;
	++_finish;
}

法三

  1. 当实现了reserve()(扩容函数)后,直接调用reserve(),再进行插入即可
  2. reserve()下面实现
void push_back(const T& x)
{
	//判断扩容
	if (_finish == _endofstorage)
	{
		reserve(capacity() == 0 ? 4 : capacity() * 2);
	}

	*_finish = x;
	++_finish;
}

reserve()

  1. reserve() 函数只在 vector 对象中没有足够的内存空间时才会进行内存分配操作,以便预留一定的内存空间,避免频繁的内存分配操作影响程序性能。
  2. 用 tmp 临时存储大小为 n 的内存空间。然后记录目前vector的所有元素个数,用for循环完成拷贝操作 (第一次增容不需要拷贝,所以有 if(_start)
  3. 最后释放掉原本_start指向的空间并更新_start,_finish,_endofstorage
void reserve(size_t n)
{
	//当_finish == _endofstorage时,此时没有多余内存空间,进行增容
	if (_finish == _endofstorage)
	{
		T* tmp = new T[n];
		size_t sz = size();

		if (_start)//当_start不为空进来(第一次增容不用进)
		{
			//memcpy是浅拷贝,在一些使用场景中会出现问题
			//memcpy(tmp, _start, sizeof(T) * size());//将原本存储的数据拷贝到tmp中
			for (size_t i = 0; i < sz; i++)
			{
				//T是int,一个个拷贝没有问题
				//T是string,每次调用=也是深拷贝
				tmp[i] = _start[i];
			}
			
			delete[] _start;
		}

		_start = tmp;
		_finish = _start + sz;
		_endofstorage = _start + n;
	}
}

resize()

  1. resize()允许增容+缩容,当 (n<当前大小)时进行缩容,反之扩容
  2. 缩容 直接将_finish指向n的位置即可
  3. 扩容 首先进行判断,然后将多出的空间全部赋值val resize()此时为扩容+初始化)
void resize(size_t n, const T& val = T())
{
	//缩小size
	if (n < size())
	{
		_finish = _start + n;
	}
	//扩大
	else
	{
		//判断增容
		if (n > capacity())
		{
			reserve(n);
		}

		while (_finish != _start + n)//将多出的空间赋值val
		{
			*_finish = val;
			++_finish;
		}
	}
}

insert()

  1. insert() - 向任意位置插入字符
  2. 首先断言pos不能越界,随后判断是否需要扩容
  3. 执行插入步骤:将pos位后的所有元素后移一位,插入pos,++_finish(多一个元素,需要更新大小)
  4. 返回值类型是 iterator:因为insert()需要返回一个指向插入的元素的迭代器。最后返回迭代器pos。
iterator insert(iterator pos, const T& x)
{
	//防止pos越界
	assert(pos >= _start);
	assert(pos <= _finish);

	//判读扩容
	if (_finish == _endofstorage)
	{
		//扩容后pos会失效,所以在扩容后更新pos
		size_t len = pos - _start;
		reserve(capacity() == 0 ? 4 : capacity() * 2);

		//扩容后更新pos
		pos = _start + len;
	}

	iterator end = _finish - 1;
	
	//将插入位置后面的元素后移
	while (end >= pos)
	{
		*(end + 1) = *end;
		--end;
	}
	*pos = x;//插入元素
	++_finish;

	return pos;
}

erase()

  1. erase()和insert()思路相似
  2. 首先断言防止pos越界,将pos后的所有元素前移一位,最后更新大小(_finish指针的位置)
iterator erase(iterator pos)
{
	assert(pos >= _start);
	assert(pos <= _finish);

	iterator begin = pos + 1;
	//将pos位后面的元素前移
	while (begin < _finish)
	{
		*(begin - 1) = *(begin);
		++begin;
	}

	--_finish;

	return pos;
}

完整代码

vector的模拟实现

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

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

相关文章

RabbitMQ消息队列实战(4)—— spring-boot-starter-amqp中消息的可靠性传输和确认机制

在上一篇文章中&#xff0c;笔者整理了从消息生产出来到消费结束的整个生命周期过程中&#xff0c;为了确保消息能够可靠到达或者消费&#xff0c;我们需要在哪些环节进行哪些处理&#xff0c;同时也展示了使用Java原生代码怎么样在这些环节进行处理。本文主要介绍使用spring b…

java静态代码块

在 Java中&#xff0c;每个类都有一个静态的代码块&#xff0c;用来描述类的构造函数和实例变量。在 java. util. Static中定义了一个静态代码块&#xff0c;在该代码块中&#xff0c;类的构造函数和实例变量都是不可以被修改的。 一个类包含了由它自己定义的静态代码块&#x…

【论文阅读】Self-paced Multi-view Co-training

论文下载 bib: ARTICLE{MaMeng2020SPamCo, title {Self-Paced Multi-View Co-Training}, author {Fan Ma and Deyu Meng and Xuanyi Dong and Yi Yang}, journal {J. Mach. Learn. Res.}, year {2020}, volume {21}, number {1}, numpages {1--38} }目录1.…

Kubernetes中的Calico网络

文章目录1 介绍2 环境部署3 IPIP模式3.1 测试环境3.2 ping包网络转发4 BGP模式4.1 测试环境4.2 ping网络转发5 两种模式对比1 介绍 Calico网络的大概思路&#xff0c;即不走Overlay网络&#xff0c;不引入另外的网络性能损耗&#xff0c;而是将转发全部用三层网络的路由转发来…

GPSS【实践 01】Developing a Greenplum Streaming Server Client 自定义GPSS客户端开发实例

自定义GPSS客户端开发流程1.GPSS是什么2.架构3.组件下载安装4.自定义客户端4.1 GPSS Batch Data API Service Definition4.2 Setting up a Java Development Environment4.3 Generating the Batch Data API Client Classes4.4 Coding the GPSS Batch Data Client4.4.1 Connect …

【论文笔记】Attention Augmented Convolutional Networks(ICCV 2019 入选文章)

目录 一、摘要 二、介绍 三、相关工作 卷积网络Convolutional networks&#xff1a; 网络中注意力机制Attention mechanisms in networks&#xff1a; 四、方法 1. 图像的自注意力Self-attention over images&#xff1a; 二维位置嵌入Two-dimensional Positional Enco…

redis 第一章

开始学习redis 之旅吧 关于redis 的介绍 redis 是一个开源的软件&#xff0c;可以存储结构化的数据在内存中&#xff0c;像内存数据库&#xff0c;缓存、消息中间件、流处理引擎。 redis 提供的数据结构像strings, hashes, lists, sets, sorted sets 。Redis具有内置复制、Lua…

《花雕学AI》13:早出对策,积极应对ChatGPT带来的一系列风险和挑战

ChatGPT是一款能和人类聊天的机器人&#xff0c;它可以学习和理解人类语言&#xff0c;也可以帮人们做一些工作&#xff0c;比如翻译、写文章、写代码等。ChatGPT很强大&#xff0c;让很多人感兴趣&#xff0c;也让很多人担心。 使用ChatGPT有一些风险&#xff0c;比如数据的质…

Pytorch 张量操作 Python切片操作

目录一维张量定义一维实例操作二维张量操作张量拼接-注意需要拼接的维度一定要相同广播机制更高维的演示总结YOLOv5 Focus样例参考梳理一下Pytorch的张量切片操作一维张量定义 一维向量的操作其实很像numpy一维数组&#xff0c;基本定义如下&#xff1a; 1.默认步长为1 2.起始…

HotSpot经典垃圾收集器

虽然垃圾收集器的技术在不断进步&#xff0c;但直到现在还没最好的收集器出现&#xff0c;更加不存在“万能”的收集器&#xff0c;所以我们选择的只是对具体应用最合适的收集器。 图 HotSpot中的垃圾收集器&#xff0c;连线表示可搭配使用 1 Serial收集器 是最基础、历史最悠…

第08章_面向对象编程(高级)

第08章_面向对象编程(高级) 讲师&#xff1a;尚硅谷-宋红康&#xff08;江湖人称&#xff1a;康师傅&#xff09; 官网&#xff1a;http://www.atguigu.com 本章专题与脉络 1. 关键字&#xff1a;static 回顾类中的实例变量&#xff08;即非static的成员变量&#xff09; c…

linux文件类型和根目录结构

目录 一、Linux文件类型 二、Linux系统的目录结构 1. FHS 2. 路径以及工作目录 &#xff08;1&#xff09;路径 &#xff08;2&#xff09;工作目录 一、Linux文件类型 使用ls -l命令查看到的第一个字符文件类型说明-普通文件类似于Windows的记事本d目录文件类似于Windo…

【GPT4】GPT4 创作郭德纲姜昆相声作品的比较研究

欢迎关注【youcans的 AIGC 学习笔记】原创作品 说明&#xff1a;本文附录内容由 youcans 与 GPT-4 共同创作。 【GPT4】GPT4 创作郭德纲姜昆相声作品的比较研究研究总结0. 背景1. 对 GPT4 创作的第 1 段相声的分析2. 对GPT4 创作的第 2 段相声的分析3. 对GPT4 创作的第 3 段相…

Window常用命令

一、快捷键 1、自带快捷键 序号快捷键作用1windowsGXBOX录屏2cmd >osk屏幕键盘3cmd >calc计算器4cmd >mrt恶意软件删除工具 2、浏览器快捷键 序号快捷键作用1Alt P浏览器图片下载&#xff08;来自油猴脚本&#xff09; 二、其他功能 1、解决端口占用 第一步&…

Linux安装单细胞分析软件copykat

Linux安装单细胞分析软件copykat 测试环境 Linux centos 7R 4.1.2minconda3天意云24C192GB安装步骤 新建环境 conda activate copykatconda install r-base4.1.2 安装基础软件 checkPkg <- function(pkg){return(requireNamespace(pkg, quietly TRUE))}if(!checkPkg("…

类的加载过程-过程二:Linking阶段

链接过程之验证阶段(Verification) 当类加载到系统后&#xff0c;就开始链接操作&#xff0c;验证是链接操作的第一步。 它的目的是保证加载的字节码是合法、合理并符合规范的。 验证的步骤比较复杂&#xff0c;实际要验证的项目也很繁多&#xff0c;大体上Java虚拟机需要做…

基于stable diffusion的艺术操作

下面是作者基于stable diffusion的艺术操作 得益于人工智能的强大技术 以下所有的图 绝对是整体星球上唯一的图 现在人工智能越来越强大&#xff0c;感觉将来最有可能取代的就是摄影师、中低级的程序员、UI设计师、数据分析师等&#xff0c;人们未来更多从事的职业应该是快速…

机器学习 01

目录 一、机器学习 二、机器学习工作流程 2.1 获取数据 2.2 数据集 2.2.1 数据类型构成 2.2.2 数据分割 2.3 数据基本处理 2.4 特征工程 2.4.1什么是特征工程 2.4.2 为什么需要特征工程(Feature Engineering) 2.4.3 特征工程内容 2.5 机器学习 2.6 模型评估 2.7 …

【消息队列】细说Kafka消费者的分区分配和重平衡

消费方式 我们直到在性能设计中异步模式&#xff0c;一般要么是采用pull&#xff0c;要么采用push。而两种方式各有优缺点。 pull &#xff1a;说白了就是通过消费端进行主动拉去数据&#xff0c;会根据自身系统处理能力去获取消息&#xff0c;上有Broker系统无需关注消费端的…

Windows GPU版本的深度学习环境安装

本文记录了cuda、cuDNN的安装配置。 参考文章&#xff1a; cuda-installation-guide-microsoft-windows 12.1 documentation Installation Guide :: NVIDIA cuDNN Documentation 一、cuda安装 注意事项&#xff1a; 1、cuda安装最重要的是查看自己应该安装的版本。 表格…