【C++进阶之路】vector的基本使用和模拟实现

news2024/11/24 16:29:01

前言

 作为STL的容器之一,vector的名字通常令人疑惑?在字面上,我们通常会翻译成向量,但感觉又解释不通,总觉得应该叫dynamic array翻译成动态数组/顺序表,更容易理解?那为啥呢?

我从知乎上看到这样的一个回答看起来挺有道理的~

在这里插入图片描述

  • 下面有一条评论也觉得挺有意思的,体会到STL的设计者的苦衷——并不是不想取,而是这个dynamic array这个名字已经被占了,不得已才取这个名字。

原文链接:c++里如何理解vector是动态数组,而这个单词本义是向量?为什么这么叫?

再查看文档对这个词的解释~

Vectors are sequence containers representing arrays that can change in size.
翻译: Vectors是一些表示动态数组顺序的容器。——就是顺序表的意思。

再查看一下定义~

template < class T, class Alloc = allocator<T> > class vector; 
//第一个参数是模板参数,第二个参数是空间配置器也叫内存池,这个参数我们先不做了解。

一、vector简单使用

①接口

这里的常用接口跟string的差不多我就讲个别跟string有区别的。

1.reserve

 这个reserve 只会扩容,其它情况不做处理,而其他情况string会给出一个模棱两可的答案——优化(具体看编译器的实现)。

②用法

1.内置类型

这里举一个int

vector<int> v;

2.二维

比如你要开一个二维动态数组(int)

vector<vector<int>> vv;

内存布局:
在这里插入图片描述

3.自定义类型

比如string

vector<string> vv;
string str("shun_hua");
vv.push_back(str);
//用类定义变量,再用变量进行初始化
vv.push_back(string("shun_hua"));
//用匿名对象初始化
vv.push_back("shun_hua");
//隐式类型转换直接初始化

说明:只要是类型,皆可以进行模板实例化。

③迭代器失效

我们只需记住两句话:

  • 数据的储存空间改变(常见的为扩容)可能会导致迭代器失效。

因为迭代器的底层是类似于指针的东西,当发生扩容时,指向旧空间的迭代器如果没有更新指向新空间,就会伴随着失效的问题。

比如:push_back,reserve,insert,swap

  • 移动数据可能会使迭代器失效。

比如:erase,在vs下使用过后判定为失效,再使用会直接出错,但是在Linux下,则不会。因此为了考虑平台移植性,我们统一认为迭代器会失效。

二、模拟实现

①要点说明

  • 1.为了不与库里面的vector冲突,我们需要命名空间对自己实现的类进行封装
  • 2.这里我们实现的框架是按照顺序表的数据结构进行实现的。
  • 3.为了理解,下面的接口是分开讲解的,最后我会给出源码

②基本框架

namespace my_vector
{
	template<class T>
	class vector
	{
	public:
		typedef T value_type;
		typedef const T const_value_type;
		typedef T* iterator;
		typedef const T* const_iterator;
		//迭代器的类型重定义
	private:
		iterator _begin = nullptr;
		iterator _finish = nullptr;
		iterator _end_of_storage = nullptr;
		//给缺省值,构造函数无需写初始化列表了,方便一些。
	};
}

大致框架图解:
在这里插入图片描述
有的小伙伴问了,为啥不用顺序表的标准形式实现呢?这里实现主要是为了学习库里的底层原理,所以库里是这样实现的,我们就这样写了,其实这样写也有好处,下面讲。

③迭代器

1.begin

iterator begin()
{
	return _begin;
}
const_iterator begin()const
{
	return _begin;
}

2.end

iterator end()
{
	return _finish;
}
const_iterator end()const
{
	return _finish;
}

浅浅的提一下:库里的cbegin与cend返回值只有const_iterator,且this指针经过特殊处理~

④ size

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

⑤capacity

size_t capacity()const
{
	return _end_of_storage - _begin;
}
  • size与capacity 运用的是指针减指针等于相邻元素个数,区间是左闭右开

⑥构造函数与析构函数

构造函数

1.默认构造函数

vector()
{
	
}

2.构造函数

//为了与下面的构造函数关联起来,这里就直接给出了。
void resize(size_t n, value_type val = value_type())
{
	if (n < size())
	{
		_finish = _finish + n;
	}
	else
	{
		reserve(n);
		iterator end = _begin + n;
		while (_finish != end)
		{
			*_finish = val;
			_finish++;
		}
		//这里直接对_finish进行调整,最后省去了一步操作。
	}
}
vector(size_t n, const value_type& val = value_type())
{
	resize(n, val);
}
//这个是用迭代器区间进行初始化
template<class InputIterator>
vector(InputIterator first , InputIterator last)
{
	size_t old_size = last - first;
	_begin = new value_type[old_size];
	InputIterator begin = first;
	while (begin != last)
	{
		push_back(*begin);
		begin++;
	}
}

这里有几个问题需要谈,我们先来谈第一个——value_type()

  • 这是C++语法支持的,适用于内置类型和自定义类型,对于内置类型会去调用它的默认构造,而对于内置类型也是可以的。

第二个问题,编译器不会按照我们想的去调用某个模板,而会去走最合适的模板。

//第一种写法
my_vector::vector<int> v(10,1);
//第二种写法
my_vector::vector<int> v(10u,1);
  • 其实第一种写法,我们是想走第一种构造函数的,但是这里的类型一样,更适合第二种构造,所以这里编译器会调用第二种,那这里如何解决呢?显然这里无法对迭代器进行显示声明,那我们只能走强制类型转换,也就是我们看到的第二种写法。

3.拷贝构造

vector(const vector& v)
{
	_begin = new value_type[v.size()];
	//大多数小伙伴可能会写第一种
	//memcpy(_begin, v._begin, sizeof(value_type) * v.size());
	//第二种	
	for (size_t i = 0; i < v.size(); i++)
	{
		_begin[i] = v[i];//不要小瞧这一步操作,下面细讲。
	}
	_finish = _end_of_storage = _begin + v.size();
}
//这个用到了reserve 和push_back,也比较方便
vector(const vector& v)
{
	reserve(v.capacity());
	for (auto e : v)
	{
		push_back(e);
	}
}

  • 这里涉及深拷贝的浅拷贝的问题。

这里举个例子。

vector<string> v;
v.push_back("1111");
v.push_back("2222");
vector<string> v1(v);

如果我们采用第一种写法:
在这里插入图片描述

  • 这里典型的是浅拷贝,运行结束,同一块空间析构两次,会报错的。

而第二种写法,会调用string类的赋值重载,完成深拷贝,如果你要说string类的赋值重载是浅拷贝,那是string类的问题不是我们的问题。

析构函数

~vector()
{
	delete[]_begin;
	_begin = _finish = _end_of_storage = nullptr;
}

⑦reserve

void reserve(size_t n = 0)
{
	if (n > capacity())
	{
		size_t old_size = size();
		iterator tmp = new value_type[n];
		//这里在自定义类型也会出深拷贝的浅拷贝问题
		//memcpy(_tmp, _begin, sizeof(value_type)*old_size);
		for (size_t i = 0; i < size(); i++)
		{
			tmp[i] = _begin[i];
		}
		delete[] _begin;
		_begin = tmp;
		_finish = _begin + old_size;
		_end_of_storage = _begin + n;
	}
}

⑧push_back

void push_back(const value_type & val)
{
	if (_finish == _end_of_storage)
	{
		size_t new_capacity = size() == 0 ? 4 : capacity() * 2;
		//扩容
		reserve(new_capacity);
	}
	*(_finish++) = val;
}

⑨[]

value_type& operator[](size_t pos)
{
	assert(pos < size());
	return _begin[pos]; 
}

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

	return _begin[pos];
}

⑩insert

void insert(iterator pos, const size_t val)
{
	assert(pos <= _finish && pos >= _begin);
	//是可以等于_finish的相当于尾插了
	if (_finish == _end_of_storage)
	{
		size_t rpos = pos - _begin;
		size_t new_capacity = size() == 0 ? 4 : capacity() * 2;
		//扩容
		reserve(new_capacity);
		pos = _begin + rpos;
	}
	iterator end = _finish;
	while (end != pos)
	{
		*(end) = *(end - 1);
		end--;
	}
	*pos = val;
	_finish++;
}

⑪erase

iterator erase(iterator pos)
{
	assert(pos < _finish&& pos >= _begin);
	//只能删除有效数据
	iterator cur = pos;
	while (cur != _finish)
	{
		*(cur) = *(cur + 1);
		cur++;
	}
	_finish--;
	return pos;
}

举例:删除偶数的代码

	my_vector::vector<int> v;
	v.push_back(1);
	v.push_back(2);
	v.push_back(2);
	v.push_back(3);
	v.push_back(4);
	v.push_back(5);
	for (auto e : v)
	{
		cout << e << " ";
	}
	cout << endl;
	//这是通用的代码
	my_vector::vector<int>::iterator it = v.begin();
	while (it != v.end())
	{
		if (*it % 2 == 0)
		{
			it = v.earse(it);
		}
		else
		{
			it++;
		}
	}
	//这是不具有平台移植性的代码
	my_vector::vector<int>::iterator it = v.begin();
	while (it != v.end())
	{
		if (*it % 2 == 0)
		{
			v.earse(it);
		}
		else
		{
			it++;
		}
	}
	//这是错误的代码,想想为什么。
	my_vector::vector<int>::iterator it = v.begin();
	while (it != v.end())
	{
		if (*it % 2 == 0)
		{
			v.earse(it);
		}
		it++;
	}
	for (auto e : v)
	{
		cout << e << " ";
	}
	cout << endl;

  • 因此:用返回值解决迭代器失效的问题,使代码具有了平台移植性。

⑫pop_back

void pop_back()
{
	earse(--end());
}

⑬swap

void swap(vector & x)
{
	std::swap(_begin, x._begin);
	std::swap(_finish, x._finish);
	std::swap(_end_of_storage, x._end_of_storage);
}

⑭ =

  • 这里我们还是采用现代写法
vector& operator =(vector tmp)
{
	swap(tmp);
	return *this;
}

源码

namespace my_vector
{
	template<class T>
	class vector
	{
	public:
		typedef T value_type;
		typedef const T const_value_type;
		typedef T* iterator;
		typedef const T* const_iterator;
		//迭代器
		iterator begin()
		{
			return _begin;
		}
		const_iterator begin()const
		{
			return _begin;
		}
		iterator end()
		{
			return _finish;
		}
		const_iterator end()const
		{
			return _finish;
		}
		vector(const vector& v)
		{
			_begin = new value_type[v.size()];
			//这里也会发生深拷贝的浅拷贝现象
			/*memcpy(_begin, v._begin, sizeof(value_type) * v.size());*/
			for (size_t i = 0; i < v.size(); i++)
			{
				_begin[i] = v[i];
			}
			_finish = _end_of_storage = _begin + v.size();
		}
		vector(size_t n, const value_type& val = value_type())
		{
			resize(n, val);
		}
		template<class InputIterator>
		vector(InputIterator first , InputIterator last)
		{
			size_t old_size = last - first;
			_begin = new value_type[old_size];
			InputIterator begin = first;
			int i = 0;
			while (begin != last)
			{
				push_back(*begin);
				begin++;
			}
		}
		//这个比较简单
		//vector(const vector& v)
		//{
		//	reserve(v.capacity());
		//	for (auto e : v)
		//	{
		//		push_back(e);
		//	}
		//}
		~vector()
		{
			delete[]_begin;
			_begin = _finish = _end_of_storage = nullptr;
		}
		size_t size() const
		{
			return _finish - _begin;
		}
		size_t capacity()const
		{
			return _end_of_storage - _begin;
		}
		void reserve(size_t n = 0)
		{
			if (n > capacity())
			{
				size_t old_size = size();
				iterator tmp = new value_type[n];
				//这里在自定义类型会出大坑
				//memcpy(_tmp, _begin, sizeof(value_type)*old_size);
				for (size_t i = 0; i < size(); i++)
				{
					tmp[i] = _begin[i];
				}
				delete[] _begin;
				_begin = tmp;
				_finish = _begin + old_size;
				_end_of_storage = _begin + n;
			}
		}
		void push_back(const value_type & val)
		{
			if (_finish == _end_of_storage)
			{
				size_t new_capacity = size() == 0 ? 4 : capacity() * 2;
				//扩容
				reserve(new_capacity);
			}
			*(_finish++) = val;
		}
		value_type& operator[](size_t pos)
		{
			assert(pos < size());
			return _begin[pos]; 
		}
		const_value_type& operator[](size_t pos)const
		{
			assert(pos < size());

			return _begin[pos];
		}
		void insert(iterator pos, const size_t val)
		{
			assert(pos <= _finish && pos >= _begin);
			if (_finish == _end_of_storage)
			{
				size_t rpos = pos - _begin;
				size_t new_capacity = size() == 0 ? 4 : capacity() * 2;
				//扩容
				reserve(new_capacity);
				pos = _begin + rpos;
			}
			iterator end = _finish;
			while (end != pos)
			{
				*(end) = *(end - 1);
				end--;
			}
			*pos = val;
			_finish++;
		}
		iterator erase(iterator pos)
		{
			assert(pos < _finish&& pos >= _begin);
			iterator cur = pos;
			while (cur != _finish)
			{
				*(cur) = *(cur + 1);
				cur++;
			}
			_finish--;
			return pos;
		}
		//尾删
		void pop_back()
		{
			earse(--end());
		}
		void swap(vector & x)
		{
			std::swap(_begin, x._begin);
			std::swap(_finish, x._finish);
			std::swap(_end_of_storage, x._end_of_storage);
		}
		//赋值
		vector& operator =(vector tmp)
		{
			swap(tmp);
			return *this;
		}
		//value_type()这里匿名算是调用默认构造,对缺省参数进行初始化
		//1.对内置类型,C++对其做了升级,有对应的默认构造
		//2.对自定义类型,会去调用其默认构造。
		void resize(size_t n, value_type val = value_type())
		{
			if (n < size())
			{
				_finish = _finish + n;
			}
			else
			{
				reserve(n);
				iterator end = _begin + n;
				while (_finish != end)
				{
					*_finish = val;
					_finish++;
				}
			}
		}
	private:
		iterator _begin = nullptr;
		iterator _finish = nullptr;
		iterator _end_of_storage = nullptr;
	};
}

总结

 今天的分享就到这里了,如果觉得文章不错,点个赞鼓励一下吧!我们下篇文章再见

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

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

相关文章

【Java】微服务项目的部署

微服务项目的部署 准备Centos安装 Docker镜像加速检查加速器是否生效 下载docker-compose方法1 curl方法2 pip方法3 直接下载released 用docker-compose部署中间件导入项目安装jdk maven git设置idea内存减小jar启动占用内存增加idea可使用内存 本文参考 https://gitee.com/…

音乐怎么转换成wav格式?分享这五个方法给大家!

在音频编辑和处理过程中&#xff0c;将音乐文件转换为WAV格式是一种常见需求。WAV格式以其无损音质和广泛的兼容性而受到许多人的喜爱。下面介绍了五种常用的方法&#xff0c;帮助您将音乐文件转换为WAV格式&#xff0c;其中方法一使用记灵在线工具。 方法一&#xff1a;记灵在…

非常规自增自减

非常规自增自减 目录 一&#xff0e; 概述二&#xff0e; 例题 一&#xff0e; 概述 在C语言的单目操作符中有&#xff08;自增&#xff09;和–&#xff08;自减&#xff09;这两个运算符。假设有变量i&#xff0c;我想让变量i加上1&#xff0c;那么我们会写成ii1这样的形式。…

自旋锁与开关中断临界区的区别

自旋锁和开关中断临界区都是用于保护共享资源的机制&#xff0c;但它们的实现方式和使用场景有所不同。 自旋锁主要是用于多核CPU上的并发编程中&#xff0c;它通过不断地检查锁的状态来等待锁的释放&#xff0c;从而避免了线程的阻塞。当一个线程需要访问共享资源时&#xff…

VTK 三维模型 体绘制 关于环境光、漫反射、镜面反射

光源: 1):环境光:环境光是一种低强度的光,由光线经过周围环境表面多次反射后形成的,利用环境光可以描述一块区域的亮度,通常在场 景中,环境光的颜色是一个常量. 2):太阳光:即定向光源,特点是从无穷远出发射光线,光线是平行的,光线强度不会随着距离衰减. 3):点光源:在有限空间…

maven安装和换源

1. 安装(17条消息) maven历史版本下载和安装_beiback的博客-CSDN博客 安装 maven历史版本仓库 下载 Index of /dist/maven/maven-3 (apache.org)https://archive.apache.org/dist/maven/maven-3/ 选择对应版本-binaries/-zip 解压即可使用 2.换源 (17条消息) 手把手教你更改…

【读书笔记】从实模式到保护模式

计算机语言 x86汇编语言&#xff1a;从实模式到保护模式&#xff08;操作系统引导课&#xff09; 原书作者李忠 用电表示数据 寄存器的作用&#xff1a;具有记忆功能的器件。锁存可以通过下面的开关控制&#xff0c;平时开关为空&#xff0c;按下开关之后&#xff0c;将输入锁…

PLEX如何搭建个人局域网的视频网站

Plex是一款功能非常强大的影音媒体管理系统&#xff0c;最大的优势是多平台支持和界面优美&#xff0c;几乎可以在所有的平台上安装plex服务器和客户端&#xff0c;让你可以随时随地享受存储在家中的电影、照片、音乐&#xff0c;并且可以实现观看记录无缝衔接&#xff0c;手机…

VScode——NPM脚本窗口找不到

一、问题描述&#xff08;NPM终端在任务栏左侧找不到&#xff09; VScode&#xff08;Visual Studio Code&#xff09;版本&#xff1a;1.79.2 二、解决办法 第一步&#xff1a;通过设置/用户设置/扩展/MPM更改NPM默认配置&#xff0c;如下图所示&#xff1a; 第二步&#xff…

[java安全]CommonsCollections6

文章目录 【java安全】CommonsCollections6**测试环境**前言分析TiedMapEntry注意点一注意点二POC调用栈 【java安全】CommonsCollections6 测试环境 3.1-3.2.1&#xff0c;jdk1.7,1.8 前言 之前我们分析了CommonsCollections1 LazyMap利用链&#xff0c;但是在java 8u71以…

2023 年中回顾:珍惜当下,锻炼身体

文章大纲 过去几十年的经验&#xff1a;人类的悲喜并不相同马太效应破圈&#xff1a;健康的身体写博客这件事&#xff1a;价值导向参考文献 距离上次阶段性回顾仅仅过去半年&#xff0c;感觉整个IT 行业天翻地覆慨而慷了。 时光荏苒&#xff0c;我自己也在芯片领域深耕了365天&…

swagger不可用

swagger不可用 问题描述问题处理测试环境外网域名测试环境内网域名思考 总结 问题描述 上周swagger还没啥问题&#xff0c;这周一测试突然和我说&#xff0c;测试环境的swagger都用不了了&#xff0c;然后开始找原因 问题处理 测试环境外网域名 一直跳这个弹窗&#xff0c;百度…

学习AJAX

AJAX &#x1f680; HTTP请求报文响应报文 &#x1f684; express框架&#x1f6ac; express基本使用 &#x1f692; 原生AJAX&#x1f6ac; GET.HTML&#x1f6ac; POST.HTML&#x1f6ac; JSON.HTML&#x1f6ac; nodemon工具可以帮助重启服务&#x1f6ac; IE缓存问题&#…

MIT 6.829 -- L0 Background: Single-Link Communication

MIT 6.829 -- L0 Background: Single-Link Communication 前言ProblemModulation(调制) & Demodulation(解调)FramingError DetectionError RecoveryARQ Shared Media Access总结 本课程为MIT 6.829 计网课程,课程对应官网链接: Computer Networks Lecture Notes 本节对应…

操作系统——虚拟内存管理

文章目录 一、虚拟内存中的几种地址1、逻辑地址2、线性地址3、逻辑地址转线性地址4、线性地址转物理地址 二、进程与内存1、内核空间和用户空间2、内存映射3、进程内存分配与回收 早期程序直接运行在物理内存上&#xff0c;直接操作物理内存&#xff0c;这种方式存在几个问题&a…

【Python】selenium项目实战:从12306网站获取特定时间段二等座有票的车次

文章目录 一、项目背景二、页面查找1、查询条件2、定位有二等座的元素3、定位有二等座的车次信息4、CtrlF检验xpath查找的车次 三、代码实现 一、项目背景 工具&#xff1a; pythonpycharmselenium 12306网址&#xff1a; https://kyfw.12306.cn/otn/leftTicket/init?linktyp…

【GESP】2023年06月图形化四级 -- 密码合规检测

密码合规检测 【题目描述】 网站注册需要有用户名和密码,默认小猫角色和白色背景,编写程序以检查用户输入密码的有效性。 (1)合法的密码只能由a-z之间26个字母(字母不区分大小写)、0-9之间10个数字以及!@#$四个特殊字母构成。 (2)密码最短长度:6个字符,密码最大长…

通用技术 自动化测试与持续集成方案

目录 前言&#xff1a; 传统接口测试 接口测试自动化 接口自动化的持续集成 前言&#xff1a; 在现代软件开发中&#xff0c;自动化测试和持续集成是两个不可或缺的环节。自动化测试可以提高测试效率、减少人工错误&#xff0c;并确保软件的质量。持续集成则可以帮助开发团…

18.JavaWeb-JWT(登录、鉴权)

1.CSRF跨站请求伪造 跨站请求伪造&#xff08;英语&#xff1a;Cross-site request forgery&#xff09;&#xff0c;也被称为 one-click attack 或者 session riding&#xff0c;通常缩写为 CSRF 或者 XSRF&#xff0c; 是一种挟制用户在当前已登录的Web应用程序上执行非本意的…

SOCKET编程基本原理

测试模拟环境 python 3.8—对应Pycharm 专业版 for linux 本文采用的环境是vmware + ubantu_64 先简单配置一下netstat工具 sudo apt-get install net-tools netstate基本用法 sudo netstat -at | grep 1234 // |管道函数 grep匹配正则表达式移除端口用Kill就可以了 SOCK…