自定义vector的实现

实现前需要思考的一个问题

为什么需要将空间的申请与对象的构建分开

查看vector的模板参数时可以看到其有第三个参数是空间适配器allocator,查找其对外提供的成员函数不难发现它的实现逻辑是将空间的申请与对象的构建分开的,为什么呢?不弄清楚这个问题显然我们是无法自定义实现一个vector的。
在这里插入图片描述

我们都知道,对于vector而言,其有一个预留空间,函数是reserve。

这是因为STL中存放的是大量元素,如果每次创建一个对象就申请一次空间,这样的话效率非常低下。所以可以直接一次性申请大片空间,然后在申请的空间上进行对象的构建,这样效率更快。

代码实现

#include <iostream>
#include <memory>
#include <algorithm>

using namespace std;

template<typename T>
class Vector{
	public:
		typedef T* iterator; //让我们写的这个类型也能通过迭代器进行访问
		Vector()
		:_start(nullptr)
		 ,_finish(nullptr)
		 ,_end_of_storage(nullptr)
		{
			
		}
		~Vector();
		void push_back(const T& value);
		void pop_back();
		int size() const;
		int capacity() const;

		//迭代器指针
		iterator begin(){
			return _start;
		}

		iterator end(){
			return _finish;
		}
	private:
		void reallocate();//重新分配内存,动态扩容要用的
	private:
		//进行空间申请与释放,以及对象构建与销毁的类
		static std::allocator<T> _alloc;

		T* _start;          //指向数组中的第一个元素
		T* _finish;         //指向最后一个实际元素之后的那个元素
		T* _end_of_storage; //指向数组本身之后的位置
};

//静态成员必须在类外进行初始化
template <typename T>
std::allocator<T> Vector<T>::_alloc;

//重新分配内存,动态扩容要用的
template <typename T>
void Vector<T>::reallocate(){
	//1、获取新空间大小
	//先存下来原来的容量
	int oldCapacity = capacity();
	//然后再将新的空间按两倍的标准进行扩容
	//如果oldCapacity为0,那么新空间大小为1,否则就按两倍扩容
	int newCapacity = 2 * oldCapacity > 0 ? 2*oldCapacity : 1;
	//2、申请新的对应大小的空间
	T* _ptmp = _alloc.allocate(newCapacity);
	//将老空间里面的元素拷贝到新的空间里面来
	if(_start){//判断老空间里面是否存在元素
		//若有数据,那么执行拷贝
		//copy(_start,_finish,_ptmp); 这个函数有缺陷
		//我们让其在未初始化的空间上拷贝对象
		uninitialized_copy(_start, _finish, _ptmp);
		//拷贝完先执行销毁对象的操作
		while(_finish != _start){
			//老的空间上的对象一个个进行销毁
			_alloc.destroy(--_finish);
		}
		//然后销毁老的空间,也就是回收老的空间
		_alloc.deallocate(_start,oldCapacity);
	}
	//三个指针之前是指向老的空间,然后扩容之后需要将三个指针与新的空间产生联系
	_start = _ptmp;
	_finish = _start + oldCapacity;
	_end_of_storage = _start + newCapacity;
}

template <typename T>
Vector<T>::~Vector(){
	if(_start){
		while(_finish!=_start){
			_alloc.destroy(--_finish);
		}
		_alloc.deallocate(_start,capacity());
	}
}

//插入元素
template <typename T>
void Vector<T>::push_back(const T& value){
	//首先判断vector是不是满的
	if(size() == capacity()){
		//如果满了,那么扩容
		reallocate();
	}
	//否则没满
	//那么当小于容量大小的时候,往vector里面送值
	if(size() < capacity()){
		//把新的value对象送到vector的末尾
		//然后_finish++继续指到最后一个实际元素的下一个位置
		_alloc.construct(_finish++,value);
	}
}

template <typename T>
void Vector<T>::pop_back(){
	//判断是否为空,不为空才删除一个末尾元素
	if(size() > 0){
		//_finish先减减来到实际元素的位置然后再执行删除
		_alloc.destroy(--_finish);
	}
}

//记录元素个数
template <typename T>
int Vector<T>::size() const{
	//因为指针是被STL的迭代器封装过的,
	//所以直接进行四则运算就可以得到正常的数值
	return _finish - _start;
}

//记录容量
template <typename T>
int Vector<T>::capacity() const{
	return _end_of_storage - _start;
}

template <typename Container>
void printCapacity(const Container& con){
	cout << "con.size()" << con.size()  << endl;
	cout << "con.capacity()" << con.capacity() << endl;
}

void test(){
	Vector<int> number;
	printCapacity(number);

	cout << endl;

	number.push_back(1);
	printCapacity(number);

	cout << endl;
	number.push_back(2);
	printCapacity(number);

	cout << endl;
	number.push_back(3);
	printCapacity(number);

	cout << endl;
	number.push_back(4);
	printCapacity(number);

	cout << endl;
	number.push_back(5);
	printCapacity(number);

	for(auto& elem : number){
		cout << elem << " ";
	}
	cout << endl;
}

int main(){
	test();
	return 0;
}

总结

其实根据vector的源码可以大概实现上面的效果,真正复杂的位置是空间配置器类alloc的实现,上述代码中我们是直接调用的该类的各种API。

如果有时间的话其实可以好好研读一下alloc的源码,看它是如何为我们的容器分配空间和回收空间的,最好是读一读《STL源码剖析》这本书,会对STL有更深刻的体会。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

ETCD 未授权访问实战案例

1、发现 etcd 未授权。 https://xxx200:2379/v2/keys 2、尝试在etcd里查询管理员的token&#xff0c;然后使用该token配合kubectl指令接管集群。 proxychains ./etcdctl --insecure-transportfalse --insecure-skip-tls-verify --endpointshttps://xxx0:2379/ get / --prefix…

算法通关村第十六关—滑动窗口经典问题(白银)

滑动窗口经典问题 一、最长子串专题 1.1 无重复字符的最长子串 LeetCode3给定一个字符串s&#xff0c;请你找出其中不含有重复字符的最长子串的长度。例如&#xff1a; 输入&#xff1a;s"abcabcbb" 输出&#xff1a;3 解释&#xff1a;因为无重复字符的最长子串是…

在Windows中安装MinGW

1、下载 github下载https://github.com/niXman/mingw-builds-binaries/releases 或官网下载https://www.mingw-w64.org/downloads/ 2、选择x86_64-12.1.0-release-posix-seh-rt_v10-rev3 3、解压到当前文件夹 解压之后&#xff0c;可以移动到自己喜欢的文件夹 &#xff0c;复…

1月12日1月15日代码随想录路经总和从中序和后序遍历构造二叉树

112.路经总和 给你二叉树的根节点 root 和一个表示目标和的整数 targetSum 。判断该树中是否存在 根节点到叶子节点 的路径&#xff0c;这条路径上所有节点值相加等于目标和 targetSum 。如果存在&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 叶子节点 …

msvcr120.dll丢失是什么意思,msvcr120.dll丢失怎样修复

msvcr120.dll是一个重要的动态链接库&#xff08;Dynamic Link Library&#xff0c;DLL&#xff09;文件&#xff0c;其中包含了Microsoft Visual C Redistributable for Visual Studio 2013的组件之一。当系统或应用程序提示该DLL文件缺失时&#xff0c;可能会导致应用程序无法…

电商物流查询:未来的发展方向

在电商日益繁荣的时代&#xff0c;物流信息查询不仅关乎消费者体验&#xff0c;更影响着电商运营的效率。快速、准确地追踪物流信息至关重要。本文将简述物流信息快速追踪的价值&#xff0c;并重点介绍固乔快递查询助手这一高效查询工具及其批量查询功能。 一、物流信息快速追踪…

gitLab创建项目无分支,本地新建module提交gitLab教程

第一&#xff1a; gitLab上创建好空项目 第二&#xff1a; 拉取到本地 本地新建module里面的代码拷到文件夹里(把里面的git文件拷到module里面也可以的) 第三&#xff1a; 默认分支为master&#xff0c;本地新建分支release1.0.0 以及 个人的分支feature/1.0.0-*** 新建分支&a…

基于springboot体育场馆运营管理系统源码

基于springboot体育场馆运营管理系统源码330 -- MySQL dump 10.13 Distrib 5.7.31, for Linux (x86_64) -- -- Host: localhost Database: springboot3cprm -- ------------------------------------------------------ -- Server version 5.7.31/*!40101 SET OLD_CHARACT…

Flink 处理函数(1)—— 基本处理函数

在 Flink 的多层 API中&#xff0c;处理函数是最底层的API&#xff0c;是所有转换算子的一个概括性的表达&#xff0c;可以自定义处理逻辑 在处理函数中&#xff0c;我们直面的就是数据流中最基本的元素&#xff1a;数据事件&#xff08;event&#xff09;、状态&#xff08;st…

论文笔记(三十九)Learning Human-to-Robot Handovers from Point Clouds

Learning Human-to-Robot Handovers from Point Clouds 文章概括摘要1. 介绍2. 相关工作3. 背景3.1. 强化学习3.2. 移交模拟基准 4. 方法4.1. Handover Environment4.2. 感知4.3. 基于视觉的控制4.4. 师生两阶段培训 (Two-Stage Teacher-Student Training) 5. 实验5.1. 模拟评估…

智能光栅光片显微成像技术的LabVIEW解决方案

智能光栅光片显微成像技术的LabVIEW解决方案 在生物医学研究中&#xff0c;高效的成像技术对于捕捉细胞内罕见和复杂事件至关重要。智能光栅光片显微技术&#xff08;smartLLSM&#xff09;的出现&#xff0c;代表了LabVIEW软件在高端成像领域的革命性应用&#xff0c;这项技术…

P1563 [NOIP2016 提高组] 玩具谜题————C++

目录 [NOIP2016 提高组] 玩具谜题题目背景题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 样例 #2样例输入 #2样例输出 #2 提示 解题思路Code运行结果 [NOIP2016 提高组] 玩具谜题 题目背景 NOIP2016 提高组 D1T1 题目描述 小南有一套可爱的玩具小人&#xff0c;它…

免费学习鸿蒙(HarmonyOS)开发,一些地址分享

HarmonyOS万物互联&#xff0c;从华为一系列的操作来看已经与iOS、Android形成三足鼎立之势了。 根据《澎湃新闻》的报道&#xff0c;已有23所985高校和46所211高校加入了鸿蒙班的行列&#xff0c;合计达到了69所国内一流高校。通过鸿蒙班的设立&#xff0c;高校可以为学生提供…

4 - JdbcTemplate

spring 框架如何处理对数据库的操作呢? 1. 基本介绍 文档&#xff1a;JdbcTemplate APIs : /spring-framework-5.3.8/docs/javadoc-api/index.html JdbcTemplate 是 Spring 提供的访问数据库的技术。可以将 JDBC 的常用操作封装为模板方法 已经提供了特别多的 API 2. 使用…

【ubuntu】docker中如何ping其他ip或外网

docker中如何ping其他ip或外网 示例图&#xff1a; 运行下面命令&#xff1a; docker run -it --namehei busybox看情况需要加权限 sudo&#xff0c;即&#xff1a; sudo docker run -it --namehei busyboxping 外网 ping -c 4 www.baidu.comping 内网 ping -c 4 192.168.…

steam游戏搬砖项目还能火多久?

最近放假回到老家&#xff0c;见了不少亲戚朋友&#xff0c;大家不约而同都在感叹今年大环境不好&#xff0c;工作不顺&#xff0c;生意效益不好&#xff0c;公司状况不佳&#xff0c;反问我们生意如何&#xff1f;为了让他们心里好受一点&#xff0c;我也假装附和道:也不咋地&…

汽车研发测试大全

车研发中需要做的试验&#xff0c;这些试验都是保证我们的车能安全、稳定、可靠行驶的必要条件。主要包含以下内容&#xff1a; 一、整车试验项目 1.1整车可靠性试验 1.2 NVH试验 1.3 HVAC试验 1.4 EMC试验 1.5 化学分析试验 1.6 整车道路性能试验 二、零部件试验项目 …

node的下载、安装、配置

下载&#xff1a; 官网下载&#xff1a;Node.js 左右两个都可以&#xff1a; 往期版本&#xff1a; ​​​​​​https://registry.npmmirror.com/binary.html?pathnode/v16.18.0/ 其中的16.18.0可以修改成你需要的版本 安装&#xff1a; 打开cmd&#xff1a; 输入以下指令…

精品基于Uniapp+springboot车辆充电桩缴费管理系统管理系统App-地图

《[含文档PPT源码等]精品基于Uniappspringboot充电桩管理系统App》该项目含有源码、文档、PPT、配套开发软件、软件安装教程、项目发布教程、包运行成功&#xff01; 软件开发环境及开发工具&#xff1a; 开发语言&#xff1a;Java 后台框架&#xff1a;springboot、ssm 安…

【STM32单片机】迷宫游戏设计

文章目录 一、主要功能二、软件设计三、实验现象联系作者 一、主要功能 本项目使用STM32F103/F407单片机控制器&#xff0c;TFTLCD触摸屏、按键等。 主要功能&#xff1a; 系统运行后&#xff0c;TFTLCD显示游戏界面&#xff0c;可按下KEY_UP键进入游戏&#xff1b; 系统内置…
最新文章