• C++ vector容器的介绍与使用


    一、vector简介

    std::vector 是 C++ 标准模板库 (STL) 中的一个动态数组容器。允许存储元素(可以使用任何数据类型作为其元素类型)集合,并能够动态调整其大小。

    特点:

    1. 动态大小:与常规数组不同,vector 可以在运行时改变大小。

    2. 连续存储:vector 保证其元素在内存中是连续存储的,可以使用指针或迭代器遍历其元素。

    3. 访问:可以使用索引运算符 [] 快速访问 vector 中的元素,与常规数组类似,此外还有 at() 方法提供带边界检查的访问。

    4. 性能:向 vector 的末尾添加/删除元素(使用 push_back 或 pop_back)通常是快速的。但在 vector 的中间或开始位置插入/删除元素需要移动多个元素,相对较慢。

    5. 内存:随着元素的添加,在当前分配的内存不足以容纳更多的元素时,vector 会重新分配内存(这可能导致之前的迭代器、指针和引用失效)

    6. 函数:vector 提供了一系列函数来管理其元素。

    本文主要参考:https://cplusplus.com/reference/vector/vector/

    关于代码的相关解释也可以在该网站中查询。

    本文主要以示例代码为演示,结合函数说明来对vector的相关用法进行阐述。

    二、初始化

    1. 代码示例

    1. #include
    2. #include
    3. using namespace std;
    4. int main() {
    5. vector v1;
    6. v1.push_back("hello");
    7. v1.push_back("world");
    8. for(int i=0; isize(); i++)cout<" "; cout<
    9. // 输出为:hello world
    10. vector<int> v2(5,99);
    11. for(int i=0; isize(); i++)cout<" "; cout<
    12. // 输出为:99 99 99 99 99
    13. vector<int> v3({0, 1, 2});
    14. vector<int> v4 = {3, 4, 5};
    15. for(int i=0; isize(); i++)cout<" "; cout<
    16. // 输出为:0 1 2
    17. for(int i=0; isize(); i++)cout<" "; cout<
    18. // 输出为:3 4 5
    19. int arr[] = {1, 2, 3, 4, 5, 6};
    20. vector<int> v5(arr,arr+5);
    21. for(int i=0; isize(); i++)cout<" "; cout<
    22. // 输出为:1 2 3 4 5
    23. vector<int> v6(v5.begin(),v5.end()-2);
    24. for(int i=0; isize(); i++)cout<" "; cout<
    25. // 输出为:1 2 3
    26. vector<int> v7(v5);
    27. for(int i=0; isize(); i++)cout<" "; cout<
    28. // 输出为:1 2 3 4 5
    29. vector<int> v8;
    30. v8 = v7;
    31. for(int i=0; isize(); i++)cout<" "; cout<
    32. // 输出为:1 2 3 4 5
    33. cout << &v5 << endl << &v7 << endl << &v8 << endl;
    34. // 输出三个地址,可以发现地址并不相同
    35. // 说明v5、v7、v8虽然内容相同,但却是两个不同的对象,位于内存中的不同位置。
    36. return 0;
    37. }

    注:

    v.begin()指向 vector 中 第一个元素 的迭代器
    v.end()  指向 vector 中 最后一个元素之后的位置 的迭代器(并不指向 vector 中的任何元素)

    2.函数说明

    1. vector();
    2. // 默认构造函数:创建一个空的 vector。
    3. // 如上述代码 vector v1;
    4. vector(size_type n);
    5. // 创建一个包含 n 个元素的 vector
    6. vector(size_type n, const value_type& value);
    7. // 创建一个包含 n 个元素,每个元素的值 value 的 vector。
    8. // vector v2(5,99);
    9. vector(initializer_list il);
    10. // 使用初始化列表il 来初始化 vector
    11. // vector v3({0, 1, 2});
    12. vector(InputIterator first, InputIterator last);
    13. // 使用初始化列表创建
    14. // vector v5(arr,arr+5);
    15. // vector v6(v5.begin(),v5.end()-2);
    16. vector(vector& x);
    17. // 复制 vector
    18. // vector v7(v5);
    19. // 上面这句代码的效果等价于 vector v7 = v5;

    2. 容量

    1. 代码示例

    1. #include
    2. #include
    3. using namespace std;
    4. int main() {
    5. vector<int> v1(2,7);
    6. for(auto t : v1)cout<< t <<" "; cout << endl;
    7. // 输出为:7 7
    8. cout<< "size:" << v1.size() << endl;
    9. cout << "capacity:" << v1.capacity() << endl;
    10. // 输出为:size:2
    11. // 输出为:capacity:2
    12. v1.push_back(8);
    13. cout<< "size:" << v1.size() << endl;
    14. cout << "capacity:" << v1.capacity() << endl;
    15. // 输出为:size:3
    16. // 输出为:capacity:4
    17. cout<< "max_size:" << v1.max_size() << endl;
    18. // 输出视情况而定,本机此处输出为:max_size:2305843009213693951
    19. v1.resize(1);
    20. for(auto t : v1)cout<< t <<" "; cout << endl;
    21. // 输出为:7
    22. cout<< "size:" << v1.size() << endl;
    23. cout << "capacity:" << v1.capacity() << endl;
    24. // 输出为:size:1
    25. // 输出为:capacity:4
    26. v1.resize(5,99);
    27. for(auto t : v1)cout<< t <<" "; cout << endl;
    28. // 输出为:7 99 99 99 99
    29. cout<< "size:" << v1.size() << endl;
    30. cout << "capacity:" << v1.capacity() << endl;
    31. // 输出为:size:5
    32. // 输出为:capacity:5
    33. v1.reserve(9);
    34. for(auto t : v1)cout<< t <<" "; cout << endl;
    35. // 输出为:7 99 99 99 99
    36. cout<< "size:" << v1.size() << endl;
    37. cout << "capacity:" << v1.capacity() << endl;
    38. // 输出为:size:5
    39. // 输出为:capacity:9
    40. v1.shrink_to_fit();
    41. for(auto t : v1)cout<< t <<" "; cout << endl;
    42. // 输出为:7 99 99 99 99
    43. cout<< "size:" << v1.size() << endl;
    44. cout << "capacity:" << v1.capacity() << endl;
    45. // 输出为:size:5
    46. // 输出为:capacity:5
    47. cout<< v1.empty() << endl;
    48. // 输出为:0
    49. cout<< "size:" << v1.size() << endl;
    50. cout << "capacity:" << v1.capacity() << endl;
    51. // 输出为:size:5
    52. // 输出为:capacity:5
    53. v1.resize(0);
    54. cout<< v1.empty() << endl;
    55. // 输出为:1
    56. cout<< "size:" << v1.size() << endl;
    57. cout << "capacity:" << v1.capacity() << endl;
    58. // 输出为:size:0
    59. // 输出为:capacity:5
    60. return 0;
    61. }

    2.函数说明

    1. size_type size();
    2. // vector中包含元素的数量。
    3. // size_type is an unsigned integral type。
    4. size_type capacity();
    5. // 返回vector已分配存储空间的大小,该值大于等于 size()。当此容量耗尽并且需要更多容量时,容器会自动扩展它(重新分配其存储空间)。
    6. size_type max_size();
    7. // vector可以容纳的最大元素数量,由系统或库的限制决定,是非常大的值。
    8. void resize(n);
    9. void resize(n, value);
    10. // 改变vector的大小,使其包含 n 个元素。当 n 大于当前大小时,添加 n - size() 个新元素,值为 value ;如果 n 小于当前大小,则将删除最后的 size() - n 个元素。resize 可能会导致 vector 的内存重新分配(但不一定,我测试了一下,没有看到capacity()的变化)。
    11. bool empty();
    12. // 判断vector的size()是否为0
    13. void reserve(n);
    14. // 请求更改capacity,如果 n 大于当前矢量容量,则该函数会导致容器重新分配其存储,将其容量增加到 n(或更大)。如果 n 小于或等于当前容量,则该函数不会产生任何效果。使用 reserve 可以提高效率,特别是知道最终需要存储多少元素时。
    15. void shrink_to_fit();
    16. // 请求容器减小其capacity以适应其size。

    3. 元素访问

    1. 代码示例

    1. #include
    2. #include
    3. #include
    4. using namespace std;
    5. int main() {
    6. vector<int> v1;
    7. v1.push_back(100);
    8. v1.push_back(99);
    9. cout<< v1[0] << endl;
    10. // 输出为:100
    11. cout<< v1.at(1) << endl;
    12. // 输出为:99
    13. v1.resize(6,1);
    14. for(auto t : v1) cout << t <<" "; cout << endl;
    15. // 输出为:100 99 1 1 1 1
    16. cout << v1.front() << " " << v1.back() << endl;
    17. // 输出为:100 1
    18. v1.front() = 88;
    19. v1.back() = 123;
    20. cout << v1.front() << " " << v1.back() << endl;
    21. // 输出为:88 123
    22. vector<int>::iterator it;
    23. for (it = v1.begin(); it != v1.end(); it++)
    24. cout << *it <<" ";
    25. cout << endl;
    26. // 输出为:88 99 1 1 1 123
    27. int* p = v1.data();
    28. for(int i=0; isize(); i++, p++) cout << *p <<" "; cout << endl;
    29. // 输出为:88 99 1 1 1 123
    30. int* q = v1.data();
    31. memset(q, 0, v1.size() * sizeof(int));
    32. for(auto t : v1) cout << t <<" "; cout << endl;
    33. // 输出为:0 0 0 0 0 0
    34. return 0;
    35. }

    2.函数说明

    1. //——————以vector v为例——————
    2. [] 运算符
    3. // 访问下标为n的元素可以使用v[n]访问
    4. reference at(n);
    5. // 访问下标为n的元素
    6. // 返回值类型 reference 是容器元素的引用类型,此处的reference类型为int
    7. reference front();
    8. // 返回对vector中第一个元素的引用
    9. reference back();
    10. // 返回对vector中最后一个元素的引用
    11. value_type* data();
    12. // 该函数返回一个指向容器中第一个元素的指针。这是一种直接访问底层数组的方法。
    13. // 此处的返回值类型 value_type* 为 int* 类型。
    14. // for (vector::iterator it = v.begin(); it != v.end(); ++it)
    15. // 使用迭代器输出每个元素

    4. 修改

    1. 代码示例

    1. #include
    2. #include
    3. using namespace std;
    4. class P {
    5. private:
    6. string name;
    7. int age;
    8. public:
    9. P(string str, int n){
    10. name = str;
    11. age = n;
    12. }
    13. void print()
    14. {
    15. cout << name << " " << age << endl;
    16. }
    17. };
    18. int main() {
    19. vector<int> v1;
    20. v1.resize(5,99);
    21. for(auto t : v1) cout << t <<" "; cout << endl;
    22. // 输出为:99 99 99 99 99
    23. v1.pop_back();
    24. v1.pop_back();
    25. for(auto t : v1) cout << t <<" "; cout << endl;
    26. // 输出为:99 99 99
    27. v1.assign({1,2,3,4,5});
    28. for(auto t : v1) cout << t <<" "; cout << endl;
    29. // 输出为:1 2 3 4 5
    30. vector<int> v2(v1.begin(), v1.begin() + 3);
    31. for(auto t : v2) cout << t <<" "; cout << endl;
    32. // 输出为:1 2 3
    33. v1.swap(v2);
    34. for(auto t : v1) cout << t <<" "; cout << endl;
    35. // 输出为:1 2 3
    36. for(auto t : v2) cout << t <<" "; cout << endl;
    37. // 输出为:1 2 3 4 5
    38. //------------以下使用class P ------------
    39. vector

      v;

    40. v.emplace_back("ubuntu", 66);
    41. v.push_back(P("windows", 99));
    42. for(auto temp : v) temp.print();
    43. // 输出为:
    44. // ubuntu 66
    45. // windows 99
    46. v.emplace(v.begin()+1, "computer", 100);
    47. for(auto temp : v) temp.print();
    48. // 输出为:
    49. // ubuntu 66
    50. // computer 100
    51. // windows 99
    52. v.insert(v.begin()+2, {P("aaa", 1),P("bbb", 2)});
    53. v.insert(v.begin()+4, P("ccc", 3));
    54. for(auto temp : v) temp.print();
    55. // 输出为:
    56. // ubuntu 66
    57. // computer 100
    58. // aaa 1
    59. // bbb 2
    60. // ccc 3
    61. // windows 99
    62. v.insert(v.begin(), v.begin()+2,v.begin() + 5);
    63. for(auto temp : v) temp.print();
    64. // 输出为:
    65. // aaa 1
    66. // bbb 2
    67. // ccc 3
    68. // ubuntu 66
    69. // computer 100
    70. // aaa 1
    71. // bbb 2
    72. // ccc 3
    73. // windows 99
    74. v.erase(v.begin(), v.begin() + 4);
    75. for(auto temp : v) temp.print();
    76. // 输出为:
    77. // computer 100
    78. // aaa 1
    79. // bbb 2
    80. // ccc 3
    81. // windows 99
    82. vector

      ::iterator it;

    83. it = v.begin();
    84. v.erase(it + 3);
    85. for(auto temp : v) temp.print();
    86. // 输出为:
    87. // computer 100
    88. // aaa 1
    89. // bbb 2
    90. // windows 99
    91. v.clear();
    92. cout << "size:" << v.size() << " capacity:" << v.capacity() << endl;
    93. // 输出为:size:0 capacity:12
    94. return 0;
    95. }

    2.函数说明

    1. void push_back(value_type&& val);
    2. // 在 vector 的末尾添加一个新元素,(当新的 vector 大小超过当前 vector 的 capacity 时,会自动重新分配存储空间)
    3. // 与其功能类似的还有emplace_back()函数
    4. // 比如向vector末尾插入一个整数100:
    5. // v.push_back(100);
    6. void emplace_back(Args&&... args);
    7. // 在 vector 末尾插入一个新元素,类似于push_back()
    8. // Args&&... args 为 基本数据类型参数,或者类的构造函数参数
    9. // emplace_back 是直接在 vector 的末尾就地构造一个元素,而不是先创建一个元素然后复制或移动它到 vector 中。
    10. // push_back 是用于添加已经创建的对象或临时对象到 vector 的末尾。
    11. // emplace_back 的效率会更高一些。
    12. void pop_back();
    13. // 删除 vector 的最后一个元素,将 size 减小 1。
    14. // pop_back()用于删除最后一个元素,注意与back()区别,back()用来访问最后一个元素。
    15. void clear();
    16. // 删除所有元素,使size为0;
    17. void swap (vector& x);
    18. // 交换两个 vector 对象的内容.
    19. // 这是一个非常高效的操作,不涉及实际数据的复制,只是交换内部指针、大小和容量。
    20. // std::vector vec1 = {1, 2, 3};
    21. // std::vector vec2 = {4, 5, 6, 7};
    22. // vec1.swap(vec2); //vec1: {4, 5, 6, 7}, vec2: {1, 2, 3}
    23. iterator erase (const_iterator position);
    24. // 从vector中删除某个元素
    25. iterator erase (const_iterator first, const_iterator last);
    26. // 从vector中删除[first,last)元素
    27. // 函数返回值为一个指向被函数调用擦除的 最后一个元素之后的元素的新位置
    28. // vector vec = {1, 2, 3, 4, 5};
    29. // vec.erase(vec.begin() + 2); // 删除第三个元素,vec现在是{1, 2, 4, 5}
    30. // vector vec1 = {1, 2, 3, 4, 5};
    31. // vec1.erase(vec1.begin() + 1, vec1.begin() + 4); // 删除第二个到第四个元素,vec现在是{1, 5}
    32. void assign(size_type n, const value_type& val);
    33. // assign会清除容器的当前内容,并将其替换为指定的新内容。
    34. // 清除vector,并使用 n 个值为 value 的元素 替换。
    35. // 比如使用 10 个 0 填充,v.assign(10, 0);
    36. void assign (InputIterator first, InputIterator last);
    37. // 从另一个容器或数组的给定范围中的元素替换 vector 的内容。
    38. // vector source = {1, 2, 3, 4, 5};
    39. // vector dest1,dest2;
    40. // dest1.assign(source.begin(), source.begin() + 3); // dest1: {1, 2, 3}
    41. // dest2.assign(source.begin(), source.end() - 3); // dest2: {1, 2}
    42. void assign (initializer_list il);
    43. // std::vector vec;
    44. // vec.assign({7, 8, 9, 10}); // vec: {7, 8, 9, 10}
    45. iterator insert(const_iterator position, const T& value);
    46. iterator insert(const_iterator position, size_type n, const T& value);
    47. // 当 n 缺省时:在指定位置插入一个元素,返回值为指向插入元素的迭代器类型 iterator
    48. // vector vec = {1, 2, 4};
    49. // vec.insert(vec.begin() + 2, 3); // vec: {1, 2, 3, 4}
    50. // 当 插入元素数量 n 指定时:在指定位置插入 n 个元素,返回值为指向插入的第一个元素的迭代器类型 iterator
    51. // vector vec = {1, 2, 4};
    52. // vec.insert(vec.begin() + 1, 2, 999); // vec: {1, 999, 999, 2, 4}
    53. // 插入操作需要从插入位置开始,向后移动所有元素,所以其时间复杂度为O(n)。
    54. iterator insert(const_iterator position, InputIterator first, InputIterator last);
    55. // 从范围插入元素:从另一个容器或数组的指定范围插入元素。
    56. // vector vec = {1, 2, 4};
    57. // int arr[] = {101, 102, 103};
    58. // vec.insert(vec.begin() + 1, arr, arr + 3); // vec1: {1, 101, 102, 103, 2, 4}
    59. iterator insert(const_iterator position, std::initializer_list il);
    60. // 从初始化列表插入元素。
    61. // vector vec = {1, 2, 4};
    62. // vec.insert(vec.begin() + 1, {101, 102, 103}); // vec: {1, 101, 102, 103, 2, 4}
    63. iterator emplace (const_iterator position, Args&&... args);
    64. // position 指定了插入元素的位置
    65. // Args&&... args 为 基本数据类型参数,或者类的构造函数参数
    66. // emplace() 和 insert()类似,都是在指定位置插入元素,但emplace()效率会更高一些
    67. // vector vec = {1, 2, 4};
    68. // vec.emplace(vec.begin() + 2, 3); // 在第三个位置插入3,vec: {1, 2, 3, 4}
    69. // vector> matrix;
    70. // matrix.emplace(matrix.begin(), vector {1, 2, 3}); // 在matrix的开始处插入一个vector

    如有不当或错误之处,恳请您的指正,谢谢!!!

  • 相关阅读:
    Android实现设置界面
    准入控制器(Admission Controller):ResourceQuota,ImagePolicyWebhook
    HTTP+ 加密 + 认证 + 完整性保护 =HTTPS(HTTPS 安全通信机制)
    WPF的数据绑定
    电动汽车交流充电桩系统的设计方案
    都说了别用BeanUtils.copyProperties,这不翻车了吧
    Java单例模式实现,一次性学完整
    在Linux中掌握不同的命令,让创建文件变得易如反掌
    网络编程--sockaddr 与 sockaddr_in
    实现领域驱动设计 - 使用ABP框架 - 创建实体
  • 原文地址:https://blog.csdn.net/qq_44667259/article/details/133473271