Loading

「学习笔记」vector

本文并不是 vector 的入门教程。

定义

std::vector 是封装动态数组的顺序容器。

vector 通常占用多于静态数组的空间,因为要分配更多内存以管理将来的增长。如果元素数量已知,可以使用 reserve() 函数提前分配内存。

操作函数

由于 vector 大家比较熟悉了,这里给大家带来一些其他不太常用的操作。

1. at(x)

访问元素,相当于 [],但是 at() 会进行越界判断,如果越界,会返回异常(即 return 3;),程序停止。
常数比 [] 稍大,即速度稍慢,但是调试的时候会比较方便。由于 [] 不会进行越界处理,且 DEV-C++ 有程序保护,所以你会访问到一个奇怪的元素,在 OJ 上提交,最终导致答案错误,而不是 RE,会对查错产生干扰。

2. resize(x, y)

这个函数与 reserve 函数很像,都是分配空间的。
resize() 函数还会改变容器大小,并进行赋值,resize(n, 1) 就是将 vector 的长度和空间改为 \(n\),且复制元素为 \(1\);但是 reserve 并不会赋值,因此 vector 里面是随机的元素。
强烈建议:如果想进行赋值操作,最好是先用 clear() 函数清空 vector,在使用 resize() 来定长赋值。

3. swap()

交换两个 vector 中的元素,但是是常数复杂度,不是 \(O_n\) 的。
在写长链剖分优化 DP,合并 DP 数组时有用过,因为复杂度比较优秀。

4. emplace()emplace_back()

它们的功能相当于 insert()push_back(),但是也有区别。
就拿 emplace_back()push_back() 举例:
push_back():向容器中加入一个右值元素(临时对象)时,首先会调用构造函数构造这个临时对象,然后需要调用拷贝构造函数将这个临时对象放入容器中。原来的临时变量释放。这样造成的问题就是临时变量申请资源的浪费。
emplace_back():引入了右值引用,转移构造函数,在插入的时候直接构造,只需要构造一次即可。
简化一点:push_back() 会先在插入时构造一个临时变量,再拷贝进入 vector,而 emplace_back() 则直接在插入时构造并放入 vector 中,没有拷贝的过程。
因此,emplace_back() 会比 push_back() 快,但也正是由于构造函数使用的区别,两者在一些情况下的使用并不一样。

(1) 一般情况下直接插入单个元素

vector<int> a;
a.push_back(1);
a.emplace_back(1);

两者的用法是一样的。

(2) 当变量类型为 pair

vector<pair<int, int> > a;
a.push_back({1, 2});
a.emplace_back(1, 2);

由于 emplace_back() 是直接构造,因此,只需要按照元素的顺序依次插入即可(元素要一一对应!),而 push_back() 则是将 {1, 2} 放入一个临时的 pair 类型的变量中,再拷贝进去。

(3) 当变量类型是结构体类型时

struct node {
	int u, v, w;
};

vector<int> a;

a.push_back(node{1, 2, 3});
a.emplace_back(1, 2, 3);

push_back() 函数没有任何问题,但是emplace_back() 就无法通过编译。
遇到这种情况,我们需要为这个结构体写一个构造函数,才可以使用 emplace_back(),像这样

struct node {
	int u, v, w;
	
	node(int a, int b, int c) {
		u = a, v = b, w = c;
	}
	
// 或者这样
//	node(int a, int b, int c) : u(a), v(b), w(c) {};
};

vector<int> a;

a.push_back(node{1, 2, 3});
a.emplace_back(1, 2, 3);

这样就是对的了。

5. 输入元素

这是一个慢但是看着很高级的写法。

vector<int> a;
a.resize(10);
for (auto &i : a) {
	cin >> i;
}

后记

关于 vector,我个人觉得这是个非常好用的 STL,方便,但是,这正因为是 STL,因此会带有常数,很可能被卡。
尽管如此,我还是很喜欢 vector
此外,关于 emplace 这个东西,dequequeue 等别的 STL 容器也有,用法都是一样的,但是,请不要搞混了。
emplace() 对应 insert 或者 push 再或者一些特殊 STL 的插入 (例如 map)
emplace_back() 对应 push_back()
emplace_front() 对应 push_front()
map 中还有 emplace_hint() 函数,用法与 emplace() 差不多。

posted @ 2023-06-25 22:23  yi_fan0305  阅读(141)  评论(2编辑  收藏  举报