问题缘起
今天在调试程序时,发现一个量经过循环后会无端增大。最后排查到是使用vector的resize时,对其理解有一定偏颇。经过修改后才得以正常运行。
回家后,查找c++primer 书籍中的resize()函数用法,重新学习一番,在此做一小结并避免以后继续有类似的坑。
resize()用法
1. resize()不适用于array容器;
2. c.resize(n) 调整c的大小为n个元素。若n<c.size(),则多出的元素被丢弃。若n>c.size(),则对*新* (请注意:是“新”)元素进行初始化;
3. c.resize(n,t) 调整c的大小为n个元素。任何新添加的元素都初始化为t.
如下面代码片段:
#include <vector>
#include <iostream>
int main()
{
std::vector<int> ivec = {1, 2, 3};
std::cout << "Before resize:" << std::endl;
for (auto i : ivec)
std::cout << i << " ";
std::cout << std::endl;
ivec.resize(5);
std::cout << "After resize:"<< std::endl;
for (auto i : ivec)
std::cout << i << " ";
return 0;
}
输出结果:
图1 执行结果
当时我在自己的程序中,由于忽略了resize是对新的数据才进行值初始化,而认为resize是对所有 值进行了值初始化。进而造成了程序运行异常。我的本意是将其resize并达到所有值为默认初始化的目的(我的容器元素构造了默认构造函数)。此时并无法达到此目的。要想达到此目的,可增加一行:
#include <vector>
#include <iostream>
int main()
{
std::vector<int> ivec = {1, 2, 3};
std::cout << "Before resize:" << std::endl;
for (auto i : ivec)
std::cout << i << " ";
std::cout << std::endl;
//增加一行,使用clear()函数将元素清楚,进而使用resize(),
//可使得所有值默认初始化
ivec.clear();
ivec.resize(5);
std::cout << "After clear and then resize:" << std::endl;
for (auto i : ivec)
std::cout << i << " ";
return 0;
}
执行结果:
图2 先clear() 后resize()结果
此时便达到了所有值都为默认初始化的结果。
猜测
当resize后,容器原来的capacity()可能小于现有resize(),那原来数据是否是利用std::move函数拷贝到新的位置了呢?还是直接拷贝?猜测大概率是利用move,因为原来数据后面不会再用了
总结
- 可以使用resize()函数改变顺序容器大小(除array外);
- 当想将容器值设为默认值,并且还想改变大小时,可以先将容器利用clear()函数释放其原来的值,并再使用resize()函数改变大小。
-2020.07.23 今天胖五发射了火星探测器。今天买了互助网去莫干山游玩的票,需要休息一下了。开心~