【STL容器】vector

news2024/9/23 16:16:39

文章目录

  • 前言
  • vector
    • 1.1 vector的定义
    • 1.2 vector的迭代器
    • 1.3 vector的元素操作
      • 1.3.1 Member function
      • 1.3.2 capacity
      • 1.3.3 modify
    • 1.4 vector的优缺点


前言

vector是STL的容器,它提供了动态数组的功能。
注:文章出现的代码并非STL库里的源码,只是对源码的简单化处理


vector

1.1 vector的定义

vector的成员变量是3个迭代器,用来管理一段线性连续空间

具体如下图:

在这里插入图片描述

1.2 vector的迭代器

vector的迭代器是指针,且属于随机迭代器
vector所管理的是一个连续线性的空间,因此无论类型如何,普通的指针就可以很好的解决问题。

注:
随机迭代器,在支持双向移动的基础上,支持前后位置的比较、随机存取、直接移动n个距离。
在这里插入图片描述

模拟如下:

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

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

1.3 vector的元素操作

1.3.1 Member function

构造函数
在这里插入图片描述
这里我们先不管空间配置器(allocator),以及explicit关键字.
简化上面的3个红框的部分

//1.
vector();
//2.
vector(size_t n, const T& t = T());
//3.
template<class InputIterator>
vector(InputIterator first, InputIterator last);

简化后1和2就容易理解了,但可能对于3还有些困惑
它的使用如下:
在这里插入图片描述
简化实现如下:

template<class InputIterator>
vector(InputIterator first, InputIterator last)
{
	while(first != last)
	{
		this->push_back(*first);
		++first;
	}
}

赋值运算符重载
对于赋值运算符重载函数,有一种现代的写法

//现代写法
void swap(vector<T>& tmp)
{
	std::swap(_start, tmp._start);
	std::swap(_finish, tmp._finish);
	std::swap(_end_of_storage, tmp._end_of_storage);
}
vector<T>& operator=(vector<T> tmp)
{
	swap(tmp);
	return *this;
}
//传统写法
vector<T>& operator=(const vector<T>& v)
{
	if(this != &v)
	{
		clear();
		detele[] _start;
		iterator tmp = new T[v.capacity()]
		_start = tmp;
		_finish = tmp + v.size();
		_end_of_storage = tmp + v.capacity();
		for(size_t i = 0; i < v.size(); i++)
		{
			_start[i] = v[i];
		}
	}
	return *this;
}

我们浅浅分析一下,
传统的赋值逻辑是先将原空间删除,再开辟一个新空间,然后将数据慢慢拷贝过去。
现代的赋值逻辑,operator=的形参tmp不是引用,是通过拷贝构造得到的局部对象,然后通过swap(), 将tmp与this的数据进行交换,这样this指向了它想要的空间,而tmp指向了旧空间,又因为tmp是局部对象,等operator=函数结束时,tmp会自动调用析构函数,释放tmp指向的空间,也就是原来的旧空间。
相比原来的旧空间,不但减少了代码,更减少了错误。

注:有些人在用传统方法写赋值运算符重载时,会犯一个严重的错误,就是使用memcpy对原来的数据进行拷贝。

错误:memcpy(_start, v._start, sizeof(T)*v.size());
正确:for(size_t i = 0; i < v.size(); i++)
{ _start[i] = v[i]; }

这是比较典型的值拷贝问题,如果vector存储的类型T是内置类型,没什么事,但如果是自定义类型, 指针则会出错
在这里插入图片描述


1.3.2 capacity

在这里插入图片描述

上面的问题也容易出现在reserve
reserve()
作用:开辟空间到n,但如果n < capacity,则不会变。总结一句,只接受变大。

void reserve(size_t n)
		{
			
			if (n > capacity())
			{
				size_t len = size();
				iterator tmp = new T[n];
				for (size_t i = 0; i < len; i++)
				{
					tmp[i] = _start[i];
				}
				delete[] _start;
				_start = tmp;
				_finish = _start + len;
				_endofstorage = _start + n;
			}
		}

resize
强制将size改到n

  1. 如果sz > n,则删除数据
  2. 如果sz < n,则扩容,并填入t
void resize(size_t n, const T t = T())
		{
			if (n < size())
			{
				_finish = _start + n;
			}
			else
			{
				if (n > capacity()) reserve(n);
				for (iterator i = _finish; i < _start + n; i++)
				{
					*i = t;
				}
				_finish = _start + n;
			}
		}

1.3.3 modify

在这里插入图片描述

这里介绍一些insert和erase

erase()
在这里插入图片描述
模拟:

iterator erase(iterator pos)
{
    assert(pos >= _start && pos < _finish);
    iterator it = pos;
    while (it != _finish)
    {
        *it = *(it + 1);
        ++it;
    }
    --_finish;
    return pos;
}
iterator erase(iterator first, iterator last)
{
	assert(first >= _start && first < _finish);
	assert(last >= _start && last <= _finish);
	assert(first<= last);
	iterator it = first;
	while(last != _finish)
	{
		*it = *last;
		++it, ++last;	
	}
	_finish = it;
	return it;
}

insert
在这里插入图片描述

void insert(iterator pos, const T& t)
{
	assert(pos >= _start && pos <= _finish);
	if (_finish == _endofstorage)
	{
		size_t n = capacity() == 0 ? 4 : 2 * capacity();
		reserve(n);
	}
	iterator end = _finish;
	while (end > pos)
	{
		*end = *(end - 1);
		--end;
	}
	*pos = t;
	_finish++;
}

注:在对容器的修改,一定要注意迭代器失效的问题
以erase为例
在这里插入图片描述
在某些平台,上面的代码是会报错的,比如vs编译器。
逻辑是这样的,v.erase(it)将it所对应位置的数据删除,删除后it就不应该具有再次访问的权利,因为这有非法访问的可能。对于vector这种连续的空间,删除数据后,后面的数据会往前补上。但如果是像list的这样的链表,删除后,iterator指向的将是一块已经被释放的空间。这很明显不对。因此,编译器决定将这样的迭代器判定为失效。但并不是所有的编译器都会这样做。比如linux下的g++对于上面的代码就不会报错。


1.4 vector的优缺点

std::vector 是 C++ STL 提供的一个非常有用的动态数组容器,它有许多优点和一些局限性,以下是其主要优缺点:

优点:

  1. 快速随机访问:由于元素在内存中是连续存储的,vector 允许在常量时间内进行随机访问,这意味着可以通过索引快速访问任何元素。

  2. 动态大小vector 允许在运行时动态增加或减少其大小,这使得它非常适合需要动态分配内存的情况。

  3. 自动内存管理vector 会自动处理内存的分配和释放,无需手动管理内存,这可以减少内存泄漏和其他与手动内存管理相关的问题。

  4. 迭代器支持vector 支持迭代器,你可以使用迭代器来遍历容器中的元素,执行各种操作,如查找、排序等。

  5. 元素复制和移动vector 可以容纳各种数据类型,包括内置类型和用户自定义类型。它会在容器内部复制或移动元素的副本,而不影响原始对象。

缺点:

  1. 插入和删除效率较低:在 vector 中插入或删除元素涉及到移动其他元素,因此插入和删除操作的效率较低。如果需要频繁的插入和删除操作,可能有更适合的容器,如 std::liststd::deque

  2. 内存浪费vector 会预留一定的内存容量以容纳更多的元素,这可能导致内存浪费,特别是当你不确定容器的最终大小时。可以使用 reserve() 来手动管理内存分配,但仍可能存在浪费。

  3. 固定的增长策略vector 的内部实现通常采用固定的增长策略,即每次扩容都会分配一块更大的内存并复制元素。这可能导致一些性能问题,特别是在大规模元素插入时。如果需要更高效的动态数组,可以考虑使用 std::deque 或其他容器。

  4. 不适用于复杂的插入操作:如果需要在中间或开头插入元素并保持高效性能,vector 可能不是最佳选择,因为插入元素后需要移动大量元素。

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

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

相关文章

C++ PrimerPlus 复习 第三章 处理数据

第一章 命令编译链接文件 make文件 第二章 进入c 第三章 处理数据 文章目录 C变量的命名规则&#xff1b;C内置的整型——unsigned long、long、unsigned int、int、unsigned short、short、char、unsigned char、signed char和bool&#xff1b;如何知道自己计算机类型宽度获…

Jenkins Maven pom jar打包未拉取最新包解决办法,亲测可行

Jenkins Maven pom jar打包未拉取最新包解决办法&#xff0c;亲测可行 1. 发布新版的snapshots版本的jar包&#xff0c;默认Jenkins打包不拉取snapshots包2. 设置了snapshot拉取后&#xff0c;部分包还未更新&#xff0c;需要把包版本以snapshot结尾3. IDEA无法更新snapshots包…

超炫的开关效果

超炫的开关动画 代码如下 <!DOCTYPE html> <html> <head><meta charset"UTF-8"><title>Switch</title><style>/* 谁家好人 一个按钮写三百多行样式代码 &#x1f622;&#x1f622;*/*,*:after,*:before {box-sizing: bor…

代码随想录--栈与队列-用队列实现栈

使用队列实现栈的下列操作&#xff1a; push(x) -- 元素 x 入栈pop() -- 移除栈顶元素top() -- 获取栈顶元素empty() -- 返回栈是否为空 &#xff08;这里要强调是单向队列&#xff09; 用两个队列que1和que2实现队列的功能&#xff0c;que2其实完全就是一个备份的作用 impo…

产教融合 | 力软联合重庆科技学院开展低代码应用开发培训

近日&#xff0c;力软与重庆科技学院联合推出了为期两周的低代码应用开发培训课程&#xff0c;来自重庆科技学院相关专业的近百名师生参加了此次培训。 融合研学与实践&#xff0c;方能成为当代数字英才。本次培训全程采用线下模式&#xff0c;以“力软低代码平台”为软件开发…

聚焦真实用例,重仓亚洲,孙宇晨畅谈全球加密新格局下的主动变革

「如果能全面监管&#xff0c;加密行业仍有非常大的增长空间。目前加密用户只有 1 亿左右&#xff0c;如果监管明朗&#xff0c;我们可以在 3-5 年内获得 20-30 亿用户。」—— 孙宇晨 9 月 14 日&#xff0c;波场 TRON 创始人、火币 HTX 全球顾问委员会成员孙宇晨受邀出席 20…

RHCSA的一些简单操作命令

目录 1、查看Linux版本信息 2、ssh远程登陆 3、解析[zxlocalhost ~]$ 与 [rootlocalhost ~]# 4、退出命令exit 5、su——switch user 6、打印用户所处的路径信息pwd 7、修改路径 8、输出文件\目录信息 9、重置root账号密码 10、修改主机名称 1&#xff09;临时修改…

SkyWalking入门之Agent原理初步分析

一、简介 当前稍微上点体量的互联网公司已经逐渐采用微服务的开发模式&#xff0c;将之前早期的单体架构系统拆分为很多的子系统&#xff0c;子系统封装为微服务&#xff0c;彼此间通过HTTP协议RESET API的方式进行相互调用或者gRPC协议进行数据协作。 早期微服务只有几个的情况…

三种方式部署单机版Minio,10行命令干就完了~

必要步骤&#xff1a;安装MinIO 拉取MinIO镜像 docker pull quay.io/minio/minio 创建文件挂载点 mkdir /home/docker/MinIO/data &#xff08;文件挂载点映射&#xff0c;默认是/mydata/minio/data&#xff0c;修改为/home/docker/MinIO&#xff0c;文件存储位置自行修改&…

随笔-嗨,中奖了

好久没有动笔了&#xff0c;都懒惰了。 前段时间&#xff0c;老妹凑着暑假带着双胞胎的一个和老妈来了北京&#xff0c;听着小家伙叫舅舅&#xff0c;还是挺稀奇的。周末带着他们去了北戴河&#xff0c;全家人都是第一次见大海&#xff0c;感觉&#xff0c;&#xff0c;&#…

qiankun 乾坤主应用访问微应用css静态图片资源报404

发现static前没有加我指定的前缀 只有加了后才会出来 解决方案: env定义前缀 .env.development文件中 # static前缀 VUE_APP_PUBLIC_PREFIX"" .env.production文件中 # static前缀 VUE_APP_PUBLIC_PREFIX"/szgl" settings文件是封了一下src\settings…

测试平台前端部署

这里写目录标题 一、前端代码打包1、打包命令2、打包完成后,将dist文件夹拷贝到nginx文件夹中3、重新编写default.conf4、将之前启动的容器进行停止并且删除,再重新创建容器5、制作Dockerfile二、编写Dockerfile一、前端代码打包 1、打包命令 npm run build2、打包完成后,…

Kubernetes学习篇之组件

Kubernetes学习篇之组件 文章目录 Kubernetes学习篇之组件前言概述控制平面组件(Control Plane Components)kube-apiserveretcdkube-schedulerkube-controller-managercloud-controller-manager Node 组件kubeletkube-proxy容器运行时(Container Runtime) 插件(Addons)DNSWeb界…

驱动开发,IO多路复用实现过程,epoll方式

1.框架图 被称为当前时代最好用的io多路复用方式&#xff1b; 核心操作&#xff1a;一棵树&#xff08;红黑树&#xff09;、一张表&#xff08;内核链表&#xff09;以及三个接口&#xff1b; 思想&#xff1a;&#xff08;fd代表文件描述符&#xff09; epoll要把检测的事件…

Ubuntu安装深度学习环境相关(yolov8-python部署)

Ubuntu安装深度学习环境相关(yolov8-python部署) 本文将从如下几个方面总结相关的工作过程&#xff1a; Ubuntu系统安装(联想小新pro16) 2.显卡驱动安装3.测试深度学习模型 1. Ubunut 系统安装 之前在台式机上安装过Ubuntu&#xff0c;以为再在笔记本上安装会是小菜一碟&…

Linux内核源码分析 (B.x)Linux内存布局

一、32位系统的内存布局 为什么要将进程地址空间划分成内核空间和用户空间&#xff1f; 这个和处理器的体系结构有关。比如X86分为ring0~ring3级别&#xff0c;ring0给内核空间使用&#xff0c;ring3给用户空间使用&#xff1b;同样的&#xff0c;ARMv7也是如此&#xff0c;svc…

openGauss学习笔记-70 openGauss 数据库管理-创建和管理普通表-查看表数据

文章目录 openGauss学习笔记-70 openGauss 数据库管理-创建和管理普通表-查看表数据70.1 查询数据库所有表的信息70.2 查询表的属性70.3 查询表的数据量70.4 查询表的所有数据70.5 查询字段的数据70.6 过滤字段的重复数据70.7 查询字段为某某的所有数据70.8 按照字段进行排序 o…

C++之unordered_map,unordered_set模拟实现

unordered_map&#xff0c;unordered_set模拟实现 哈希表源代码哈希表模板参数的控制仿函数增加正向迭代器实现*运算符重载->运算符重载运算符重载! 和 运算符重载begin()与end()实现 unordered_set实现unordered_map实现map/set 与 unordered_map/unordered_set对比哈希表…

python 自(3)1使用urlencode多个参数请求使用 2百度翻译post请求post无法添加路径 3百度翻译全部数据获取 4豆瓣get请

1 使用urlencode 多个参数请求使用 # 使用urlencode 多个参数请求使用 # https://www.baidu.com/s?wd周杰伦&sex男 网页 import urllib.request import urllib.parsebase_url https://www.baidu.com/s?data {wd: 周杰伦,sex: 男,sing:歌曲 }new_data urllib.par…

牛客: BM3 链表中的节点每k个一组翻转

牛客: BM3 链表中的节点每k个一组翻转 文章目录 牛客: BM3 链表中的节点每k个一组翻转题目描述题解思路题解代码 题目描述 题解思路 用一个[]int保存一组节点的val,一个快节点先遍历k个节点将节点的val顺序保存在[]int中,然后慢节点再遍历k个节点,逆序将[]int的val设置给节点的…