C++ ——— 模拟实现 vector 类

news2025/1/22 6:07:26

目录

vector 类的框架

无参数的构造函数

析构函数

获取有效数据个数

获取容量

重载 [] 运算符

可读可写版本 

只可读版本

扩容

尾插

实现迭代器

可读可写版本 

只可读版本

自定义设置size长度和内容

在任意位置插入

删除任意位置的数据

赋值重载


vector 类的框架

namespace rjj
{
	template<class T>
	
	class vector
	{
    public:    
        typedef T* iterator;

	private:
		iterator _start;
		iterator _finish;
		iterator _endofstorage;
	};
}

首先把要模拟实现的 vector 类用命名空间区分开来,以免和编译器自带的 vector 类冲突

vector 类要支持不同类型的实例化,所以使用模板来定义,这样在实例化的时候就能实例化不同的类型:vector<int>、vector<char>、vector<string> 等

_start 是指向空间起始位置的指针

_finish 是指向有效数据后一个位置的指针

_endofstorage 是指向开辟的动态空间末尾的指针


无参数的构造函数

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

析构函数

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

获取有效数据个数

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

获取容量

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

重载 [] 运算符

可读可写版本 

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

	return _start[pos];
}

只可读版本

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

	return _start[pos];
}

扩容

void reserve(size_t n)
{
	// 要保证扩容的大小大于当前容量
	if (n > capacity())
	{
		size_t sz = size();

		// 开辟新空间
		T* tmp = new T[n];

		// 拷贝并且释放旧空间
		if (_start)
		{
			// 这样会出现浅拷贝问题
			// memcpy(tmp, _start, sizeof(T) * size());
			
			// 深拷贝
			for (int i = 0; i < sz; i++)
			{
				tmp[i] = _start[i];
			}
			
			delete[] _start;
		}

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

尾插

void Push_back(const T& x)
{
	// 先判断是否需要扩容
	if (_finish == _endofstorage)
    {
	    reserve(capacity() == 0 ? 4 : capacity() * 2);
    }

	// 尾插数据
	*_finish = x;
	_finish++;
}

测试代码:


实现迭代器

可读可写版本 

iterator begin()
{
	return _start;
}

iterator end()
{
	return _finish;
}

直接使用原生指针即可实现

代码演示:

只可读版本

typedef const T* const_iterator;
const_iterator begin() const
{
	return _start;
}
const_iterator end() const
{
	return _finish;
}

自定义设置size长度和内容

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++;
		}
	}
}

当要设置的长度小于或等于 size() 时,也就是小于有效数据长度时,直接减小 _finish 的指向即可

当要设置的长度大于 size() 时,先判断是否需要扩容,在从 _finish 的位置依次往后赋值

测试代码:

小于时:

大于时:


在任意位置插入

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

	// 先判断是否需要扩容
	if (_finish == _endofstorage)
	{
		// 防止异地扩容后 pos 变成野指针,先记录长度
		size_t len = pos - _start;

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

		// 再重新赋值
		pos = _start + len;
	}

	// 指向最后一个数据的下一个位置
	iterator end = _finish;

	// 向后挪动
	while (end > pos)
	{
		*end = *(end - 1);
		end--;
	}

	// 插入数据
	*pos = x;
	_finish++;
}

插入数据就会存在会扩容情况,如果扩容的话,pos 还是指向原来的空间,只要发生了扩容就会导致 pos 变成野指针,所以要使用 pos - _start 计算出长度先存储,不论是否扩容,再重新赋值 pos

测试代码:


删除任意位置的数据

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

	// 指向 pos 位置的后一个位置
	iterator next = pos + 1;

	// 向前覆盖
	while(next < _finish)
	{
		*(next - 1) = *next;
		next++;
	}

	_finish--;

	return pos;
}

测试代码:


赋值重载

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> tmp)
{
	swap(tmp);

	return *this;
}

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

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

相关文章

02内存结构篇(D4_JVM内存分配机制)

目录 一、对象的创建 1. 类加载检查 2. 分配内存 3. 初始化零值 4. 设置对象头 32位对象头 64位对象头 5. 执行方法 二、对象大小与指针压缩 三、对象内存分配 1. 对象内存分配流程图 2. 对象栈上分配 3.3 对象在Eden区分配 3.4 大对象直接进入老年代 3.5 长期存…

异步 进程 Promise规范及应用

异步 两个或多个事件不同时存在或发生&#xff0c;区别于同步&#xff0c;同步是顺序执行从上到下&#xff0c;而异步不需要顺序执行&#xff0c;且不依赖于前面的事情是否已完成。 举例&#xff1a; //异步执行 let count 1; let timer setTimeout(function () {count…

C#树图显示目录下所有文件以及文件大小

C#树图显示目录下所有文件以及文件大小 我们在打开某个目录属性时,可以查看到有大小信息.如下图 而一个目录(文件夹)System.IO.Directory是没有FileSize或者Length属性的. 目录(文件夹)的大小是指该目录下所有子目录和所有文件大小的累加,按字节为单位. 新建窗体应用程序Get…

力扣707题——设计链表

#题目 从零开始设计链表&#xff0c;我们拆分成两次任务&#xff0c;今天先看1 ,2 ,4 #代码

机器学习09-Pytorch功能拆解

机器学习09-Pytorch功能拆解 我个人是Java程序员&#xff0c;关于Python代码的使用过程中的相关代码事项&#xff0c;在此进行记录 文章目录 机器学习09-Pytorch功能拆解1-核心逻辑脉络2-个人备注3-Pytorch软件包拆解1-Python有参和无参构造构造方法的基本语法示例解释注意事项…

天津市开通首个万兆宽带:1秒钟下载1GB文件

快科技1月21日消息&#xff0c;华为光网宣布&#xff0c;天津联通携手华为&#xff0c;日前成功为天津市北辰区柴楼金园小区部署了天津市首个万兆宽带网络。 在现场测速环节中&#xff0c;该万兆宽带网络展现出了惊人的速度——高达9429Mbps的下载速率&#xff0c;几乎跑满带宽…

html全局遮罩,通过websocket来实现实时发布公告

1.index.html代码示例 <div id"websocket" style"display:none;position: absolute;color:red;background-color: black;width: 100%;height: 100%;z-index: 100; opacity: 0.9; padding-top: 30%;padding-left: 30%; padding-border:1px; "onclick&q…

数据结构(四) B树/跳表

目录 1. LRU 2. B树 3. 跳表 1. LRU: 1.1 概念: 最近最少使用算法, 就是cache缓存的算法. 因为cache(位于内存和cpu之间的存储设备)是一种容量有限的缓存, 有新的数据进入就需要将原本的数据进行排出. 1.2 LRU cache实现: #include <iostream> #include <list>…

Spring Boot 整合 ShedLock 处理定时任务重复执行的问题

&#x1f337; 古之立大事者&#xff0c;不惟有超世之才&#xff0c;亦必有坚忍不拔之志 &#x1f390; 个人CSND主页——Micro麦可乐的博客 &#x1f425;《Docker实操教程》专栏以最新的Centos版本为基础进行Docker实操教程&#xff0c;入门到实战 &#x1f33a;《RabbitMQ》…

CSDN年度回顾:技术征途上的坚实步伐

嘿&#xff0c;时光过得可真快呀&#xff0c;就像那匹跑得飞快的白马&#xff0c;嗖的一下&#xff0c;2024 年的日历就这么悄无声息地翻到了最后一页。这会儿我回头看看在 CSDN 上度过的这一年&#xff0c;心里那叫一个感慨万千&#xff0c;满满的都是喜悦&#xff0c;就像心里…

解决 PostgreSQL 中创建 TimescaleDB 扩展的字符串错误

解决 PostgreSQL 中创建 TimescaleDB 扩展的字符串错误 在使用 PostgreSQL 数据库并尝试创建 TimescaleDB 扩展时&#xff0c;你可能会遇到一些棘手的错误。今天&#xff0c;我们就来探讨一个常见的错误信息及相应的解决方法&#xff1a; CREATE EXTENSION IF NOT EXISTS tim…

【语言处理和机器学习】概述篇(基础小白入门篇)

前言 自学笔记&#xff0c;分享给语言学/语言教育学方向的&#xff0c;但对语言数据处理感兴趣但是尚未入门&#xff0c;却需要在论文中用到的小伙伴&#xff0c;欢迎大佬们补充或绕道。ps&#xff1a;本文不涉及公式讲解&#xff08;文科生小白友好体质&#xff09;&#xff…

【ESP32】ESP32连接JY61P并通过WIFI发送给电脑

前言 手头上有个ESP32&#xff0c;发现有wifi功能&#xff0c;希望连接JY61P并通过WIFI把姿态数据发送给电脑 1.采用Arduino IDE编译器&#xff1b;需要安装ESP32的开发板管理器&#xff1b; 2.电脑接受数据是基于python的&#xff1b; 1. ESP32 连接手机WIFI #include <…

将 Docker 安装到 Windows 的 D 盘的方法

1.官网下载Docker Desktop Installer.exe&#xff0c;官网网址&#xff1a;Get Started | Docker 2.以管理员身份打开 Windows 终端 3.切换到 你Docker Desktop Installer.exe下载到的地址 4.在运行代码前&#xff0c;需要提前手动创建好D:\Program Files\Docker和D:\Program F…

DeepSeek-R1-GRPO理解

一、GRPO GRPO&#xff08;Group Relative Policy Optimization&#xff09;是一种强化学习&#xff08;Reinforcement Learning, RL&#xff09;算法&#xff0c;专门用于训练大型语言模型&#xff08;LLMs&#xff09;在复杂任务&#xff08;如数学推理、代码生成等&#xf…

Unreal Engine 5 C++ Advanced Action RPG 十章笔记

第十章 Survival Game Mode 2-Game Mode Test Map 设置游戏规则进行游戏玩法 生成敌人玩家是否死亡敌人死亡是否需要刷出更多 肯定:难度增加否定:玩家胜利 流程 新的游戏模式类游戏状态新的数据表来指定总共有多少波敌人生成逻辑UI告诉当前玩家的敌人波数 3-Survival Game M…

接口(1)

大家好&#xff0c;今天我们来看看接口的概念&#xff0c;接口跟类的使用是有一些区别的&#xff0c;接口中都是抽象方法&#xff0c;简单介绍一下后&#xff0c;我们正式来学习。 2、接口 接口就是公共的行为规范标准,大家在实现时,只要符合规范标准,就可以通用. 在java中,…

docker离线安装及部署各类中间件(x86系统架构)

前言&#xff1a;此文主要针对需要在x86内网服务器搭建系统的情况 一、docker离线安装 1、下载docker镜像 https://download.docker.com/linux/static/stable/x86_64/ 版本&#xff1a;docker-23.0.6.tgz 2、将docker-23.0.6.tgz 文件上传到服务器上面&#xff0c;这里放在…

debian中apt的配置与解析

引言 在系统使用过程中&#xff0c;我们可能会遭遇 apt update 操作出现问题&#xff0c;或者 apt upgrade 速度迟缓的情况。这往往是由于所使用软件源本身存在诸如服务器性能不佳、维护不及时等质量问题&#xff0c;同时&#xff0c;软件源服务器与我们所处地理位置的距离较远…

python创建一个httpServer网页上传文件到httpServer

一、代码 1.server.py import os from http.server import SimpleHTTPRequestHandler, HTTPServer import cgi # 自定义请求处理类 class MyRequestHandler(SimpleHTTPRequestHandler):# 处理GET请求def do_GET(self):if self.path /:# 响应200状态码self.send_response(2…