C++之动态数组vector解读

 更新时间:2025年06月03日 10:02:54   作者:zzh_zao  
这篇文章主要介绍了C++之动态数组vector解读,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

在 C++ 编程中,std::vector 是标准模板库(STL)中非常重要的容器之一。

它提供了一个动态数组的功能,能够根据需要自动调整大小,同时具备高效的内存管理和丰富的操作接口。

一、什么是 std::vector?

std::vector 是 C++ STL 中的一种序列容器,它类似于传统的数组,但具有动态扩展和收缩的能力。

与普通数组相比,std::vector 的大小可以在运行时动态变化,而普通数组的大小在定义时就已经确定,无法改变。

std::vector 的底层实现是一个连续的内存块,这使得它在随机访问元素时非常高效,类似于数组的访问速度。

二、std::vector 的基本特性

(一)动态扩展

std::vector 的最大特点是动态扩展。当向 std::vector 中添加元素,而当前分配的内存空间不足以容纳更多元素时,std::vector 会自动分配更大的内存空间,并将原有元素复制到新的内存中。

这个过程虽然涉及到内存分配和数据复制,但 std::vector 会尽量优化,通常会分配比当前需要更多的空间,以减少后续的扩展次数。

(二)随机访问

由于 std::vector 的底层是连续的内存块,因此它支持随机访问。

可以通过下标(operator[]at())快速访问任意位置的元素,时间复杂度为 O(1)。

这使得 std::vector 在需要频繁随机访问元素的场景中非常高效。

(三)内存管理

std::vector 会自动管理内存。当向 std::vector 中添加或删除元素时,它会自动调整内存的分配和释放。

此外,std::vector 提供了一些方法来控制内存的分配策略,例如 reserve() 方法可以预先分配足够的内存,从而减少动态扩展的次数,提高性能。

三、std::vector 的基本操作

(一)定义和初始化

std::vector 可以通过多种方式定义和初始化。

以下是一些常见的定义方式:

#include <vector>

// 定义一个空的 vector
std::vector<int> vec1;

// 使用初始化列表初始化
std::vector<int> vec2 = {1, 2, 3, 4, 5};

// 使用默认值初始化
std::vector<int> vec3(10, 0); // 10个元素,初始值为0

// 复制构造
std::vector<int> vec4(vec2);

// 从另一个 vector 的一部分构造
std::vector<int> vec5(vec2.begin() + 1, vec2.end() - 1);

(二)添加和删除元素

std::vector 提供了多种方法来添加和删除元素:

添加元素

  • push_back():在 std::vector 的末尾添加一个元素。
  • insert():在指定位置插入一个或多个元素。
vec1.push_back(10); // 在末尾添加一个元素
vec1.insert(vec1.begin() + 2, 20); // 在索引为2的位置插入一个元素

删除元素

  • pop_back():删除 std::vector 的最后一个元素。
  • erase():删除指定位置的一个或多个元素。
vec1.pop_back(); // 删除最后一个元素
vec1.erase(vec1.begin() + 1); // 删除索引为1的元素

(三)访问元素

std::vector 提供了多种方式来访问元素:

  • operator[]:通过下标访问元素,不进行边界检查。
  • at():通过下标访问元素,并进行边界检查,如果超出范围会抛出异常。
int value1 = vec1[0]; // 使用下标访问
int value2 = vec1.at(1); // 使用 at() 访问

(四)遍历

可以使用迭代器或基于范围的 for 循环来遍历 std::vector

// 使用迭代器遍历
for (std::vector<int>::iterator it = vec1.begin(); it != vec1.end(); ++it) {
    std::cout << *it << " ";
}

// 使用基于范围的 for 循环
for (int value : vec1) {
    std::cout << value << " ";
}

(五)大小和容量

std::vector 提供了一些方法来获取其大小和容量:

  • size():返回当前 std::vector 中的元素数量。
  • capacity():返回当前分配的内存容量(以元素数量为单位)。
  • empty():判断 std::vector 是否为空。
  • resize():调整 std::vector 的大小。
  • reserve():预先分配内存,以减少动态扩展的次数。
std::cout << "Size: " << vec1.size() << std::endl;
std::cout << "Capacity: " << vec1.capacity() << std::endl;
if (vec1.empty()) {
    std::cout << "Vector is empty" << std::endl;
}
vec1.resize(15, 0); // 调整大小为15,新元素初始化为0
vec1.reserve(20); // 预先分配20个元素的内存

四、std::vector 的应用场景

(一)动态数组

std::vector 是实现动态数组的首选容器。它可以在运行时动态调整大小,非常适合需要频繁添加或删除元素的场景。

例如,在处理动态数据集合时,std::vector 可以方便地存储和管理数据。

(二)随机访问

由于 std::vector 支持随机访问,因此在需要频繁通过下标访问元素的场景中非常高效。

例如,在实现算法时,经常需要通过下标访问数组中的元素,std::vector 可以很好地满足这一需求。

(三)内存管理

std::vector 提供了灵活的内存管理功能。

通过 reserve() 方法可以预先分配足够的内存,从而减少动态扩展的次数,提高性能。

这在处理大量数据时非常有用,可以避免频繁的内存分配和数据复制。

五、注意事项

(一)性能优化

虽然 std::vector 会自动管理内存,但在某些情况下,手动控制内存分配可以提高性能。

例如,在知道数据量的情况下,可以使用 reserve() 方法预先分配足够的内存,避免多次动态扩展。

(二)内存释放

当不再需要 std::vector 时,它会自动释放分配的内存。但如果在程序运行过程中需要释放内存,可以使用 clear() 方法清空 std::vector,但需要注意的是,clear() 只会清空元素,不会释放内存。如果需要释放内存,可以使用 shrink_to_fit() 方法。

(三)异常安全

std::vector 的某些操作可能会抛出异常,例如 at() 方法在访问超出范围的元素时会抛出 std::out_of_range 异常。在使用这些方法时,需要注意异常处理。

总结

std::vector 是 C++ STL 中非常重要的容器之一,它结合了动态数组的灵活性和数组的高效性。

通过动态扩展、随机访问和灵活的内存管理,std::vector 可以满足多种编程需求。在实际开发中,合理使用 std::vector 可以提高代码的可读性和性能。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • C++实现简易通讯录

    C++实现简易通讯录

    这篇文章主要为大家详细介绍了C++实现简易通讯录,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-07-07
  • String类的写时拷贝实例

    String类的写时拷贝实例

    下面小编就为大家带来一篇String类的写时拷贝实例。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-01-01
  • c++实现值机系统

    c++实现值机系统

    这篇文章主要为大家详细介绍了c++实现在线值机系统程序,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03
  • 浅谈C++的浅拷贝出现的错误

    浅谈C++的浅拷贝出现的错误

    下面小编就为大家带来一篇浅谈C++的浅拷贝出现的错误。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-01-01
  • 一文总结C++运算符的使用方法

    一文总结C++运算符的使用方法

    这篇文章主要为大家详细总结了C++中运算符的使用方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2023-05-05
  • 简单聊聊C++中线程的原理与实现

    简单聊聊C++中线程的原理与实现

    C++11 引入了多线程支持,提供了一套基本的线程库,包括线程、互斥量(mutex)、条件变量(condition_variable)等。这些组件可以帮助你在 C++ 程序中实现并发和多线程编程,本文就来和大家简单聊聊吧
    2023-03-03
  • C# interface与delegate效能比较的深入解析

    C# interface与delegate效能比较的深入解析

    本篇文章是对C#中interface与delegate的效能比较进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • C++实现LeetCode(241.添加括号的不同方式)

    C++实现LeetCode(241.添加括号的不同方式)

    这篇文章主要介绍了C++实现LeetCode(241.添加括号的不同方式),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C++ this原理与可变参数及友元函数友元类分步详解用法

    C++ this原理与可变参数及友元函数友元类分步详解用法

    可变参数模板(variadic templates)是C++11新增的强大的特性之一,它对模板参数进行了高度泛化,能表示0到任意个数、任意类型的参数,这篇文章主要介绍了C++ this原理与可变参数及友元函数友元类
    2022-11-11
  • 数据结构 红黑树的详解

    数据结构 红黑树的详解

    这篇文章主要介绍了数据结构 红黑树的详解的相关资料,数据结构中的二叉树查找,红黑树的讲解,需要的朋友可以参考下
    2017-07-07

最新评论