[C++初阶]vector的初步理解

news2024/12/22 11:59:27

一、标准库中的vector类

1.vector的介绍

  1. vector是表示可变大小数组的序列容器 和数组一样,vector可采用的连续存储空间来存储元素。也就是意味着可以采用下标对vector的元素进行访问,和数组一样高效。但是又不像数组,它的大小是可以动态改变的,而且它的大小会被容器自动处理。
2. 本质讲,vector使用动态分配数组来存储它的元素。当新元素插入时候,这个数组需要被重新分配大小,以增加存储空间。其做法是:分配一个新的数组,然后将全部元素移到这个数组。就时间而言,这是一个代价相对来说高的任务,因为每当一个新的元素加入到容器的时候,vector并不会每次都重新分配大小。
3. vector分配空间策略:vector会分配一些额外的空间以适应可能的增长,因为存储空间比实际需要的存储空间更大。不同的库采用不同的策略权衡空间的使用和重新分配。但是无论如何,重新分配都应该是对数增长的间隔大小,以至于在末尾插入一个元素的时候是在常数时间的复杂度完成的。
4. 因此,vector占用了更多的存储空间,为了获得管理存储空间的能力,并且以一种有效的方式动态增长。
5. 与其它动态序列容器相比(deque, list and forward_list), vector在访问元素的时候更加高效,在末尾添加和删除元素相对高效。对于其它不在末尾的删除和插入操作,效率更低。比起list和forward_list统一的迭代器和引用更好 
我们知道STL学习的三步:能用,明理,能扩展,因而我们的学习也是这个流程

2.vector的使用

记住,学stl一定一定要学会看文档,vector文档.vector在实际中非常的重要,在实际中我们熟悉常

见的接口就可以。

2.1 常用的构造函数

vector();

vector类的默认构造函数,构造一个没有元素的空容器

例如:

void test()
{
	vector<int> v;
}

需要引用头文件vector


vector(size_type n, const value_type& val = value_type());

构造一个vector类对象并用n个val初始化

例如:

void test()
{
	//vector<int> v;
	vector <int> v(5, 0);
	for (auto i : v)
	{
		cout << i << " ";
	}
}
int main()
{
	test();
	return 0;
}

运行结果:

void test()
{
	//vector<int> v;
	vector <int> v(5, 2);
	for (auto i : v)
	{
		cout << i << " ";
	}
}
int main()
{
	test();
	return 0;
}

运行结果:


拷贝构造函数

vector(const vector& x);

对于一个已有的vector容器进行拷贝,并生成一个新的容器

例如:

void test()
{
	//vector<int> v;
	vector <int> v1(5, 2);
	vector <int> v2(v1);
	for (auto i : v2)
	{
		cout << i << " ";
	}
}

运行结果:v


使用迭代器的初始化构造

Template<class InputIterator>

vector(InputIterator first, InputIterator last);

例如:

void test()
{
	//vector<int> v;
	vector <int> v1(5, 2);
	//vector <int> v2(v1);
	vector <int> v2(v1.begin(),v1.end()-1);
	for (auto i : v2)
	{
		cout << i << " ";
	}
}

运行结果v:

2.2 容量操作的接口

1.size
size_type size() const;

获取有效元素的个数

例如:

void test()
{
	//vector<int> v;
	vector <int> v1(5, 2);
	//vector <int> v2(v1);
	/*vector <int> v2(v1.begin(),v1.end()-1);*/
	//for (auto i : v2)
	//{
	//	cout << i << " ";
	//}
	cout << v1.size();
}
int main()
{
	test();
	return 0;
}

运行结果:

2.capacity

获取容量大小

size_type capacity() const;

例如:

void test()
{
	vector <int> v1(5, 2);
	cout << v1.capacity();
}

运行结果:

3.empty

判断容器是否为空

bool empty() const;

例如:

​
void test()
{
	vector <int> v1(5, 2);
	vector <int> v2;
	cout << v1.empty() << endl;
	cout << v2.empty() << endl;
}

​

运行结果:

4.resize
void resize (size_type n, value_type val = value_type());

将有效元素的个数修改为n,并且如果n大于原来的size,多出来的地方用val填充

如果没有给出val,就用0填充

例如:

void test()
{
	vector <int> v;
	cout << v.size() << endl;
	v.resize(20);
	cout << v.size() << endl;
	for (auto i : v)
	{
		cout << i << " ";
	}
	cout << endl;
	v.resize(30);
	cout << v.size() << endl;
	for (auto i : v)
	{
		cout << i << " ";
	}
	cout << endl;
	v.resize(10);
	cout << v.size() << endl;
	for (auto i : v)
	{
		cout << i << " ";
	}
}

运行结果:

5.max_size
size_type max_size() const;

返回最大大小

例如:

ps:这里push_back是尾插

void test1()
{
	vector <int> v;
	for (int i = 0; i < 100; i++) 
    v.push_back(i);
	cout << v.size() << endl;
}

运行结果:

6.reserve​​​​​
void reserve (size_type n);

改变容量大小

例如:

void test1()
{
	vector <int> v;
	cout << v.capacity() << endl;
	v.reserve(10);
	cout << v.capacity() << endl;
}

运行结果:

2.3vector的访问及其遍历

1.operator[]
reference operator[] (size_type n);
const_reference operator[] (size_type n) const;

用下标直接访问

例如:

void test()
{
	vector <int> v(5,2);
	cout << v[2] << endl;
}

运行结果:

因此我们也可以通过下标去遍历,这里就不多作演示

2.迭代器

这里大体和string类差不多

iterator begin();

const_iterator begin() const;

iterator end();

const_iterator end() const;

迭代器,用于获取容器中第一个元素的位置和最后一个元素的下一个位置

例如:

​
void test()
{
	vector <int> v = { 1,2,3,4,5,67,8,9,0,1 };
	vector<int>::iterator it = v.begin();
	while (it != v.end())
	{
		cout << *it << " ";
		++it;
	}
}

​

运行结果:

3.反向迭代器

这个整体和迭代器差不多

reverse_iterator rbegin();

const_reverse_iterator rbegin() const;

reverse_iterator rend();

const_reverse_iterator rend() const;

rbegin获取容器中最后一个元素的位置,rend获取容器中的第一个元素的前一个位置

例如:

​
void test()
{
	vector <int> v = { 1,2,3,4,5,67,8,9,0,1 };
	vector<int>::reverse_iterator rit = v.rbegin();
	while (rit != v.rend())
	{
		cout << *rit << " ";
		++rit;
	}
}

​

运行结果:

4.at
reference at (size_type n);
const_reference at (size_type n) const;

返回容器中位置n处的元素的引用

例如:

void test()
{
	vector <int> v = { 1,2,3,4,5,67,8,9,0,1 };
	cout << v.at(5) << endl;
	for (size_t i=0;i<v.size();++i)
	{
		cout << v.at(i) << " ";
	}
}

运行结果:

5.back
reference back();
const_reference back() const;

返回容器中最后一个元素的引用

例如:

void test()
{
	vector <int> v = { 1,2,3,4,5,67,8,9,0,123 };
	cout << v.back()<< endl;
}

运行结果:

6.front
reference front();
const_reference front() const;

返回容器中第一个元素的引用

例如:

void test()
{
	vector <int> v = { 1,2,3,4,5,67,8,9,0,123 };
	cout << v.front()<< endl;
}

运行结果:

2.4vector的增删查改

1.push_back();
void push_back(const value_type& val);

从容器尾部插入一个元素

例如:

void test()
{
	vector <int> v = { 1,2,3,4,5,67,8,9,0,123 };
	cout << v.back()<< endl;
	v.push_back(99999);
	cout << v.back() << endl;
}

运行结果:

2.pop_back()
void test()
{
	vector <int> v = { 1,2,3,4,5,67,8,9,0,123 };
	cout << v.back()<< endl;
    v.pop_back();
	cout << v.back() << endl;
}

运行结果:

3.find()
template <class InputIterator, class T>

InputIterator find(InputIterator first, InputIterator last, const T& val);

在两个迭代器区间寻找val并返回其所在处的迭代器

例如:

void test2()
{
	vector<int> v = { 1,2,3,4,5,6 };
	for (auto i : v)
	{
		cout << i << " ";
	}
	cout << endl;
	vector<int>::iterator it = find(v.begin(), v.end(), 3);
	*it = 30;
	for (auto i : v)
	{
		cout << i << " ";
	}
	cout << endl;
}

需要注意的是,该函数并非vector的成员函数,是标准库中的函数,多个容器共用该find函数

4.insert()
iterator insert (iterator position, const value_type& val);
void insert (iterator position, size_type n, const value_type& val);

template <class InputIterator>    
void insert (iterator position, InputIterator first, InputIterator last);

在position位置插入元素

例如:

void test2()
{
	vector<int> v = { 1,2,3,4,5,6 };
	for (auto i : v)
	{
		cout << i << " ";
	}
	cout << endl;
	vector<int>::iterator it = find(v.begin(), v.end(), 3);
	v.insert(it, 99999);
	for (auto i : v)
	{
		cout << i << " ";
	}
	cout << endl;
}

运行结果:

5.earse()
iterator erase (iterator position);
iterator erase (iterator first, iterator last);

删除position位置的元素或者 [first,last) 区间的所有元素

例如:

void test2()
{
	vector<int> v = { 1,2,3,4,5,6 };
	for (auto i : v)
	{
		cout << i << " ";
	}
	cout << endl;
	vector<int>::iterator it = find(v.begin(), v.end(), 3);
	v.erase(it);
	for (auto i : v)
	{
		cout << i << " ";
	}
	cout << endl;
}

运行结果:

6.swap()
void swap (vector& x);

交换两个vector的数据空间

例如:

void test3()
{
	vector<int> v1 = { 1,2,3,4,5,6,7,8,9,0 };
	vector<int> v2 = { 0,9,8,7,6,5,4,3,2,1 };
	v1.swap(v2);
	for (auto i : v1)
	{
		cout << i << " ";
	}
	cout << endl;
}

运行结果:

7.assign()
template <class InputIterator>  void assign (InputIterator first, InputIterator last);
void assign (size_type n, const value_type& val);

为vector指定新内容,替换其当前内容并修改size

例如:

void test2()
{
	vector<int> v = { 1,2,3,4,5,6 };
	for (auto i : v)
	{
		cout << i << " ";
	}
	cout << endl;
	v.assign(5, 4);
	for (auto i : v)
	{
		cout << i << " ";
	}
	cout << endl;
}

运行结果:

8.clear()
void clear();

从vector中删除所有元素,不改变容量大小

例如:

void test2()
{
	vector<int> v = { 1,2,3,4,5,6 };
	for (auto i : v)
	{
		cout << i << " ";
	}
	cout << "size:" << v.size();
	cout << endl;
	v.clear();
	for (auto i : v)
	{
		cout << i << " ";
	}
	cout << "size:" << v.size();
	cout << endl;
}

运行结果:


本篇是对vector的初步理解,我们只需知道会用即可

如有错误,欢迎大家在评论区指出。

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

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

相关文章

营销故事之扩大牙膏开口

职场营销故事“扩大牙膏开口”又可以说是“牙膏开口扩大1毫米”&#xff0c;为十大经典营销故事之一。某品牌的牙膏&#xff0c;包装精美&#xff0c;品质优良&#xff0c;备受顾客喜爱&#xff0c;连续10年营业额保持10%-20%的增幅。可到了第11年&#xff0c;销售业绩却停滞不…

API-正则表达式

学习目标&#xff1a; 掌握正则表达式 学习内容&#xff1a; 什么是正则表达式语法元字符修饰符 什么是正则表达式&#xff1a; 正则表达式是用于匹配字符串中字符组合的模式。在JavaScript中&#xff0c;正则表达式也是对象。 通常用来查找、替换那些符合正则表达式的文本&a…

泛微开发修炼之旅--26前端j实现手机号码验证

文章链接&#xff1a;26前端j实现手机号码验证

开关电源中强制连续FCCM模式与轻载高效PSM,PFM模式优缺点对比笔记

文章目录 前言一、连续FCCM模式优点&#xff1a;缺点&#xff1a; 二,轻载高效PSM&#xff0c;PFM优点&#xff1a;缺点: 总结 前言 今天我们来学习下开关电源中&#xff0c;强制连续FCCM模式与轻载高效PSM&#xff0c;PFM模式优缺点对比 一、连续FCCM模式 优点&#xff1a; …

安装 VisualSVN Server提示HTTP服务无法启动的问题解决

安装 VisualSVN Server 版本&#xff1a;VisualSVN-Server-5.4.0-x64 安装包在安装到一半的时候&#xff0c;弹窗提示&#xff1a;HTTP服务无法启动&#xff0c;网上找了一大堆&#xff0c;说是service里面更改用户为本地用户什么的都没用用&#xff0c;点右键也无法启动。 …

【Python实战因果推断】17_线性回归的不合理效果7

目录 Regression for Dummies Conditionally Random Experiments Dummy Variables Regression for Dummies 回归和正交化固然很好&#xff0c;但归根结底&#xff0c;你必须做出独立性假设。你必须假设&#xff0c;在考虑到某些协变量的情况下&#xff0c;干预看起来与随机分…

力扣67 二进制求和

文章目录 1. 题目链接2. 题目代码3.感受 1. 题目链接 二进制求和 2. 题目代码 class Solution { public:string addBinary(string a, string b) {vector<int> stringA;vector<int> stringB;int lengthOfA a.length();int lengthOfB b.length();for(int subscrip…

【C++】初步认识C++

1. 初识C1.1 C概念相关1.2 C发展史及其重要性1.2.1 发展史1.2.2 重要性 2. C关键字3. 命名空间4. 输入和输出 个人主页&#xff1a;C_GUIQU 归属专栏&#xff1a;【C学习】 1. 初识C 1.1 C概念相关 C语言是结构化和模块化的语言&#xff0c;适合处理较小规模的程序。 【来源】…

seq2seq+Attention机制原理介绍

一、Seq2seq的局限性 Seq2seq&#xff08;序列到序列&#xff09;模型我们在前面讲了它的原理&#xff0c;是一种广泛用于处理序列转换任务的深度学习架构&#xff0c;特别是在机器翻译、文本摘要、对话生成等应用中。然而&#xff0c;尽管seq2seq模型在某些领域取得了显著的成…

数据结构----栈和队列之队列的实现

目录 1.基本概况 2.队列组成 3.队列的实现 &#xff08;1&#xff09;队列的初始化 &#xff08;2&#xff09;队列的销毁 &#xff08;3&#xff09;队列的尾插 &#xff08;4&#xff09;队列的头删 &#xff08;5&#xff09;队列的判空 &#xff08;6&#xff09;队…

量产工具一一文字系统(三)

目录 前言 一、文字数据结构抽象 1.描述一个文字的位图 2.描述一个字库操作 3.font_manager.h 二、实现Freetype封装 1.freetype.c 三、实现文字管理 1.font_manager.c 四、单元测试 1.font_test.c 2.disp_manager.c 3.disp_manager.h 4.上机测试 前言 前面我们…

vue-router拆分音乐播放界面实战

创建项目 npm install -g pnpm pnpm create vite安装 pnpm add vue-routersrc/main.js import {createApp} from vue import ./style.css import App from ./App.vue import router from "./router/index.js";const app createApp(App) app.use(router) app.moun…

2024上半年网络工程师考试《应用技术》试题一

阅读以下说明&#xff0c;回答问题。 【说明】 MPLS基于(1)进行转发&#xff0c;进行MPLS标签交换和报文转发的网络设备称为(2)&#xff0c;构成MPLS域(MPSDomain)。位于MPLS域边缘、连接其他网络的LSR称为(3),区域内部的LSR称为核心LSR(CoreLSR)IP报文进入MPLS网络时&#xf…

微软关闭中国所有线下店,并不影响全球第一

​关注卢松松&#xff0c;会经常给你分享一些我的经验和观点。 微软没有被时代淘汰&#xff0c;时代也没有告别微软!中国市场对微软可有可无&#xff0c;即便没有中国市场&#xff0c;微软市值也在全球前三&#xff0c;这是事实!a 5月中旬&#xff0c;微软azure解散中国分部…

泛微开发修炼之旅--30 linux-Ecology服务器运维脚本

文章链接&#xff1a;30 linux-ecology服务器运维脚本

Java线程同步的特征和安全类型

一线程同步的特征 ◆不同的线程在执行以同一个对象作为锁标记的同步代码块或同步方法时&#xff0c;因为要获得这个对象的锁而相互牵制&#xff0c;多个并发线程访问同一资源的同步代码块或同步方法时。 ◆同一时刻只能有一个线程进入synchronized(this)同步代码块。 ◆当一个…

SpringBoot 通过Knife4j集成API文档 在线调试

介绍 Knife4j 是一款基于 Swagger 构建的增强型 API 文档生成工具&#xff0c;它提供了更多的定制化功能和界面优化&#xff0c;使得生成的 API 文档更加美观和易用。它可以帮助开发者快速生成和管理 API 文档&#xff0c;支持在线调试和交互。 依赖 <!--knife4j--> &…

Python容器 之 练习题

1.字符串的基本使用 # 定义一个字符串 str1, 字符串的内容为 "hello world and itcast and itheima and Python" str1 "hello world and itcast and itheima and Python" # 在字符串str1中查找 字符串 and 的下标 num str1.find(and) print(num) # 12…

【设计模式】行为型-状态模式

在变幻的时光中&#xff0c;状态如诗篇般细腻流转。 文章目录 一、可调节的灯光二、状态模式三、状态模式的核心组件四、运用状态模式五、状态模式的应用场景六、小结推荐阅读 一、可调节的灯光 场景假设&#xff1a;我们有一个电灯&#xff0c;它可以被打开和关闭。用户可以…

Promethuse-监控 Etcd

一、思路 Prometheus监控Etcd集群&#xff0c;是没有对应的exporter&#xff0c;而 由CoreOS公司开发的Operator&#xff0c;用来扩展 Kubernetes API&#xff0c;特定的应用程序控制器&#xff0c;它用来创建、配置和管理复杂的有状态应用&#xff0c;如数据库、缓存和监控系…