vector系列--可怕的迭代器失效(vector重新申请内存)

Posted noticeable

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了vector系列--可怕的迭代器失效(vector重新申请内存)相关的知识,希望对你有一定的参考价值。

vector给我们提供了很多的方便,但是偶尔也会有陷阱。当不注意的时候,就掉入其中。说到底,还是对vector的机制不够彻底掌握。

很轻松的写下这段代码:

#include<iostream>
#include<vector>
using namespace std;
int main()
{
    vector<int> v;
    v.push_back(1);

    std::vector<int>::iterator iter1 = v.begin();
    v.push_back(1);

    int n = *iter1;//shit
    cout << n << endl;
    return 0;

}

  上面的代码运行时崩溃,就是因为迭代器iter1在vector push_back新值后失效了。切记!
但是为什么会失效?
如果我先预先resize足够大,那么会如何呢?

#include<iostream>
#include<vector>
using namespace std;
int main()
{
    vector<int> v;
    v.resize(4);
    v.push_back(1);

    std::vector<int>::iterator iter1 = v.begin();
    v.push_back(1);
    int n = *iter1;
    cout << n << endl;
    return 0;

}

  上面的代码正常运行,因为之前resize了足够的空间,即迭代器不会失效。

我们就已给vector push_back两个元素为例。resize(1), resize(2), resize(3)程序还是会崩溃。还是容量不足,导致vector重新分配内存,而导致迭代器失效。

之前讲过了resize与reserve的区别,那么上面的代码如果使用reserve呢?

#include<iostream>
#include<vector>
using namespace std;
int main()
{
    vector<int> v;
    v.reserve(2);
    v.push_back(0);

    std::vector<int>::iterator iter1 = v.begin();
    v.push_back(1);
    int n = *iter1;
    cout << n << endl;
    return 0;

}

  还是以push_back两个元素为例,reserve(1)还是会导致崩溃,还是造成迭代器的失效。

接下来的问题就是vector的内存机制,当容量不够时,会申请多少内存?
还是写个程序看看:

#include<iostream>
#include<vector>
using namespace std;
int main()
{
    vector<int> v;
    for (int i = 1; i < 100; i++)
    {
        cout << "capacity:" << v.capacity() << ", " << "size" << v.size() << endl;
        v.push_back(i);

    }
    return 0;

}
//输出:
capacity:0, size0
capacity:1, size1
capacity:2, size2
capacity:3, size3
capacity:4, size4
capacity:6, size5
capacity:6, size6
capacity:9, size7
capacity:9, size8
capacity:9, size9
capacity:13, size10
capacity:13, size11
capacity:13, size12
capacity:13, size13
capacity:19, size14
capacity:19, size15
capacity:19, size16
capacity:19, size17
capacity:19, size18
capacity:19, size19
capacity:28, size20
capacity:28, size21
capacity:28, size22
capacity:28, size23
capacity:28, size24
capacity:28, size25
capacity:28, size26
capacity:28, size27
capacity:28, size28
capacity:42, size29
capacity:42, size30
capacity:42, size31
capacity:42, size32
capacity:42, size33
capacity:42, size34
capacity:42, size35
capacity:42, size36
capacity:42, size37
capacity:42, size38
capacity:42, size39
capacity:42, size40
capacity:42, size41
capacity:42, size42
capacity:63, size43
capacity:63, size44
capacity:63, size45
capacity:63, size46
capacity:63, size47
capacity:63, size48
capacity:63, size49
capacity:63, size50
capacity:63, size51
capacity:63, size52
capacity:63, size53
capacity:63, size54
capacity:63, size55
capacity:63, size56
capacity:63, size57
capacity:63, size58
capacity:63, size59
capacity:63, size60
capacity:63, size61
capacity:63, size62
capacity:63, size63
capacity:94, size64
capacity:94, size65
capacity:94, size66
capacity:94, size67
capacity:94, size68
capacity:94, size69
capacity:94, size70
capacity:94, size71
capacity:94, size72
capacity:94, size73
capacity:94, size74
capacity:94, size75
capacity:94, size76
capacity:94, size77
capacity:94, size78
capacity:94, size79
capacity:94, size80
capacity:94, size81
capacity:94, size82
capacity:94, size83
capacity:94, size84
capacity:94, size85
capacity:94, size86
capacity:94, size87
capacity:94, size88
capacity:94, size89
capacity:94, size90
capacity:94, size91
capacity:94, size92
capacity:94, size93
capacity:94, size94
capacity:141, size95
capacity:141, size96
capacity:141, size97
capacity:141, size98

  

选几个数据分析一下:
0-1-2-3-4-6-9-13-19-28-42-63-94-141

从第二项开始:
2/2+2=3
3/2+3=4
4/2+4=6
6/2+6=9
9/2+9=13
13/2+13=19
19/2+19=28
28/2+28=42
42/2+42=63
63/2+63=94
94/2+94=141

每次扩容50%
删除容器中数据的时候,缓冲区大小并不会改变,仅仅只是清除了其中的数据,只有在析构函数调用的时候vector才会自动释放缓冲区。

拓展阅读

vector迭代器失效解决方法 及 内存原理

C++中STL中vector和list的迭代器失效问题

以上是关于vector系列--可怕的迭代器失效(vector重新申请内存)的主要内容,如果未能解决你的问题,请参考以下文章

vector之二(迭代器失效与解决)

vector之二(迭代器失效与解决)

vector之二(迭代器失效与解决)

vector迭代器失效的两种情况

vector迭代器失效

C++详解stl迭代器 (vector迭代器失效问题)