[C++随笔录] vector使用

vector使用

通过上一篇文章 string的模拟实现, 其实我们就已经踏入了STL的门槛.

STL容器的大致用法是差不多的 ⇒ 那我们这篇博客就讲一点跟string类不一样的新颖的东西

初始化

跟string一样, vector可以采用下面的形式初始化

cpp 复制代码
// 默认空间, 默认初始化
vector<int> vec;

// 开10个空间, 初始化为系统默认
vector<int> vec(10);

// 开10个空间, 初始化为1
vector<int> vec(10, 1);

vector可以采用 迭代器区间初始化👇👇👇

  1. 采用同类型的迭代器区间
cpp 复制代码
vector<string> vs1;
vs1.push_back("hello world");

vector<string> vs2 (vs1.begin(), vs1.end());
for (auto e : vs2)
{
	cout << e << " ";
}
cout << endl;

*****
hello world
*****
  1. 采用不同类型的迭代器区间
cpp 复制代码
vector<char> vc(5, 'b');

vector<int> vs2(vc.begin(), vc.end());
for (auto e : vs2)
{
	cout << e << " ";
}
cout << endl;

*****
98 98 98 98 98
*****
  1. 采用数组
cpp 复制代码
int a[5] = { 1,2,3,4,5 };
vector<int> vi (a, a + sizeof(a) / sizeof(int));

for (auto e : vi)
{
	cout << e << " ";
}
cout << endl;

*****
1 2 3 4 5
*****

排序算法

vector容器本身是没有 sort算法的, 但是算法库中是有 sort算法的, 头文件是 <algorithm>

👇👇👇

cpp 复制代码
int a[8] = { 11,25,3,14,5, 6, 18,20 };
vector<int> vi (a, a + sizeof(a) / sizeof(int));

sort(vi.begin(), vi.end());

for (auto e : vi)
{
	cout << e << " ";
}
cout << endl;

*****
3 5 6 11 14 18 20 25
*****

发现, 默认是升序的, 那么如何进行降序呢?

cpp 复制代码
int a[8] = { 11,25,3,14,5, 6, 18,20 };
vector<int> vi (a, a + sizeof(a) / sizeof(int));

greater<int> gt; // greater<int> 仿函数对象
sort(vi.begin(), vi.end(), gt);

for (auto e : vi)
{
	cout << e << " ";
}
cout << endl;

*****
25 20 18 14 11 6 5 3
*****

int a[8] = { 11,25,3,14,5, 6, 18,20 };
vector<int> vi (a, a + sizeof(a) / sizeof(int));

less<int> l;
sort(vi.begin(), vi.end(), l);

for (auto e : vi)
{
	cout << e << " ";
}
cout << endl;

*****
3 5 6 11 14 18 20 25
*****

上面的 greaterless 是两个仿函数, int 是 模版参数

由于 sort是一个 函数模版 ⇒ 里面的参数是 对象

那么greater 和 less两个仿函数也要传两个对象过去, 即 gtl

⇒ 由此, 我们可以得出: sort函数默认里面放的是less仿函数的对象

由于里面传的是一个对象, 那么匿名对象也是可以的

其实, 我们也可以这样写👇👇👇

cpp 复制代码
int a[8] = { 11,25,3,14,5, 6, 18,20 };
vector<int> vi (a, a + sizeof(a) / sizeof(int));

sort(vi.begin(), vi.end(), greater<int> ());

for (auto e : vi)
{
	cout << e << " ";
}
cout << endl;

其实, 如果要排降序, 我们也可以抖个机灵反向迭代器

cpp 复制代码
int a[8] = { 11,25,3,14,5, 6, 18,20 };
vector<int> vi (a, a + sizeof(a) / sizeof(int));

sort(vi.rbegin(), vi.rend());

for (auto e : vi)
{
	cout << e << " ";
}
cout << endl;

*****
25 20 18 14 11 6 5 3
*****

reverse和resize

  1. reverse --- --- 开空间
  2. resize --- --- 开空间 + 初始化

基础不好的同学看到这里就有疑问了:

🗨️ reverse 和 resize 的功能重复了, 要一个就可以了! 开空间是最重要的, 那要一个reverse就可以. 初始化啥的, 我们后面自己弄就行了

  • 其实, 这个问题, 我们在 string的模拟实现 中就有答案!
    先看下面的一个例子👇👇👇
cpp 复制代码
vector<int> vec;
vec.reserve(10);

for (int i = 0; i < 10; i++)
{
	vec[i] = i;
}

*****
error
*****

这是因为 operator[] 的实现中, 我们有一个断言 ⇒ assert(i < _size)

这里, 我们开了10个空间 ⇒ _capacity = 10;

但 _size(数据个数) = 0;

⇒ 由于断言, 我们是不能插入数据的

如果要用 []来进行插入数据, 那么我们要用如下的方式👇👇👇

cpp 复制代码
vector<int> vec;
vec.resize(10);

for (int i = 0; i < 10; i++)
{
	vec[i] = i;
}

那我们非要用 reverse 来进行插入数据, 那么我们不应该用 [], 而是用 push_back

  1. push_back 和 _size没有关系, 不够了也只是扩容而已
  2. push_back过后, _size是会进行++的
cpp 复制代码
vector<int> vec;
vec.reserve(10);

for (int i = 0; i < 10; i++)
{
	vec.push_back(i);
}

由此, 我们可以得出:

  1. operator[] 是跟 _size 密切相关的
  2. push_back 的使用就比较宽松
  3. 并不是开好了空间, 就一定能进行插入数据的

人需在事上磨,方可立得住,方能静亦定,动亦定. --- ---- 王阳明
译:有个学生问王阳明,安静的时候我感觉很不错,思想清晰,可一遇到事情就乱了阵脚,为什么?
王阳明说:"这是你只知道静养,却没有下克己的功夫.这样一来,碰到事情就乱了阵脚. 人应该在具体的事情上磨炼自己,才能站得稳,才能静亦定,动亦定.

相关推荐
rit84324994 小时前
基于MATLAB的模糊图像复原
开发语言·matlab
fie88894 小时前
基于MATLAB的声呐图像特征提取与显示
开发语言·人工智能
未来之窗软件服务4 小时前
自己写算法(九)网页数字动画函数——东方仙盟化神期
前端·javascript·算法·仙盟创梦ide·东方仙盟·东方仙盟算法
豐儀麟阁贵4 小时前
基本数据类型
java·算法
_extraordinary_5 小时前
Java SpringMVC(二) --- 响应,综合性练习
java·开发语言
Larry_Yanan6 小时前
QML学习笔记(三十四)QML的GroupBox、RadioButton
c++·笔记·qt·学习·ui
@。1246 小时前
对于灰度发布(金丝雀发布)的了解
开发语言·前端
程序员老舅6 小时前
干货|腾讯 Linux C/C++ 后端开发岗面试
linux·c语言·c++·编程·大厂面试题
乐迪信息6 小时前
乐迪信息:基于AI算法的煤矿作业人员安全规范智能监测与预警系统
大数据·人工智能·算法·安全·视觉检测·推荐算法
程序员Aries6 小时前
自定义网络协议与序列化/反序列化
linux·网络·c++·网络协议·程序人生