如何在C++中初始化一个vector_C++ vector初始化方法汇总

C++11前初始化vector主要依赖构造函数,如指定大小或范围初始化;常见陷阱包括混淆列表初始化与大小初始化,以及未预分配空间导致频繁内存重分配影响性能。

如何在c++中初始化一个vector_c++ vector初始化方法汇总

初始化std::vector在C++中其实有很多种玩法,说白了,就是告诉这个动态数组你一开始想装些什么,或者想让它有多大。从最直接的指定大小和默认值,到C++11后方便的列表初始化,再到从另一个容器“复制”过来,每种方式都有它存在的道理和适用的场景。选择哪种,往往取决于你手头的数据情况和编码习惯。

说到vector的初始化,最常用也最直观的几种方法,在我看来,主要围绕着“数量”和“内容”这两个核心点。

一种很常见的情况是,你已经知道vector需要存放多少个元素,但暂时不关心它们的具体值,或者希望它们都有一个默认值。比如:

#include #include  // 为了string类型示例#include  // 为了输出int main() {    // 1. 指定大小,元素默认初始化(对基本类型通常是0,对类类型调用默认构造函数)    std::vector vec1(5); // 包含5个int,值都是0    std::cout << "vec1: ";    for (int x : vec1) {        std::cout << x << " ";    }    std::cout << std::endl; // 输出: 0 0 0 0 0    // 2. 指定大小并赋初始值    std::vector vec2(3, 100); // 包含3个int,值都是100    std::cout << "vec2: ";    for (int x : vec2) {        std::cout << x << " ";    }    std::cout << std::endl; // 输出: 100 100 100    // 3. C++11后的列表初始化,这玩意儿简直是福音,简洁又直观    std::vector vec3 = {"apple", "banana", "cherry"};    std::cout << "vec3: ";    for (const std::string& s : vec3) {        std::cout << s << " ";    }    std::cout << std::endl; // 输出: apple banana cherry    // 也可以直接用花括号    std::vector vec4{1.1, 2.2, 3.3};    std::cout << "vec4: ";    for (double d : vec4) {        std::cout << d << " ";    }    std::cout << std::endl; // 输出: 1.1 2.2 3.3    // 4. 从另一个vector拷贝(或者移动,但初始化时拷贝更常见)    std::vector vec5 = vec2; // vec5是vec2的副本    std::cout << "vec5 (copy of vec2): ";    for (int x : vec5) {        std::cout << x << " ";    }    std::cout << std::endl; // 输出: 100 100 100    // 5. 范围初始化:从一对迭代器指定的范围初始化    // 比如从vec3初始化一个string vector    std::vector vec6_str(vec3.begin(), vec3.end());     std::cout << "vec6_str (range init from vec3): ";    for (const std::string& s : vec6_str) {        std::cout << s << " ";    }    std::cout << std::endl; // 输出: apple banana cherry}

列表初始化(std::initializer_list)是我个人最喜欢用的方式,因为它简洁明了,特别适合在编译期就确定了所有元素的情况。它不仅能用于vector,很多标准库容器都支持,统一了初始化语法,减少了心智负担。而指定大小和默认值的方式,则更适用于需要预分配空间,或者元素内容后续会填充的场景。

立即学习“C++免费学习笔记(深入)”;

C++11之前,std::vector有哪些初始化方式?

在C++11引入std::initializer_list之前,vector的初始化确实没现在这么“花哨”,但基本的也够用。那时候,如果你想初始化一个vector,主要还是靠构造函数。

比如,最基础的就是默认构造函数,它会创建一个空的vector

std::vector myVec; // 创建一个空的vectorstd::cout << "myVec size: " << myVec.size() << std::endl; // 输出: myVec size: 0

如果你知道需要多少个元素,并且希望它们都一样,那么带大小和初始值的构造函数就派上用场了:

std::vector numbers(10, 5); // 包含10个5// 或者只指定大小,元素会默认初始化std::vector names(3); // 包含3个空字符串

还有一种很强大的方式是范围构造函数。这允许你从任何一对迭代器指定的范围中初始化vector。这意味着你可以从数组、其他vectorlist甚至文件流中读取数据来初始化。

int arr[] = {1, 2, 3, 4, 5};// 注意:std::begin和std::end是C++11引入的,但概念在C++03也可以通过指针实现std::vector fromArray(arr, arr + sizeof(arr)/sizeof(arr[0])); // 从C风格数组初始化std::vector anotherVec_old; // 假设这是C++03的方式,先push_backanotherVec_old.push_back(10.1);anotherVec_old.push_back(20.2);anotherVec_old.push_back(30.3);std::vector copiedVec(anotherVec_old.begin(), anotherVec_old.end()); // 从另一个vector初始化

坦白讲,在C++11之前,如果我想初始化一个带有特定内容的vector,但内容又不是重复的,最常见的做法是先创建一个空的vector,然后用push_back或者insert逐个添加元素。这虽然有点啰嗦,但胜在灵活。当然,如果数据源本身就是另一个容器或者一个迭代器范围,那范围构造函数无疑是最高效且优雅的选择。

初始化std::vector时常见的陷阱和性能考量是什么?

初始化vector看似简单,但里面其实藏着一些小坑和性能上的门道,尤其是在处理大量数据或者对性能有较高要求时。

一个常见的误解是关于vectorcapacitysize。当你用std::vector vec(10);初始化时,vecsize是10,capacity至少是10。但如果你是先默认构造一个空的,然后循环push_back10次,capacity可能会经历多次重新分配和拷贝。每次push_back如果导致capacity不足,vector会重新分配一块更大的内存,然后把旧内存的数据拷贝过去,再释放旧内存。这个过程开销不小,尤其是在循环次数多的时候。

所以,如果预先知道vector最终会包含多少个元素,使用reserve()预留空间是个好习惯:

std::vector myNumbers;myNumbers.reserve(1000); // 预留1000个元素的空间,此时size仍为0for (int i = 0; i < 1000; ++i) {    myNumbers.push_back(i); // 这里就不会频繁地重新分配内存了}

另一个容易混淆的点是列表初始化和带有大小参数的构造函数。

std::vector v1(5); // 5个0std::vector v2{5}; // 1个5

看到没?v1创建了5个默认初始化的整数(通常是0),而v2却创建了一个只包含一个元素5的vector。这是因为当只有一个参数且类型可以转换为initializer_list时,编译器会优先选择initializer_list构造函数。这在某些情况下可能会导致意想不到的行为,特别是当你期望的是一个特定大小的vector时。所以,在使用单参数初始化时,务必清楚你想要的是什么。

性能方面,拷贝

以上就是如何在C++中初始化一个vector_C++ vector初始化方法汇总的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1477809.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月19日 02:21:35
下一篇 2025年12月19日 02:21:42

相关推荐

  • 如何在C++中检查一个文件是否存在_C++文件存在性检查方法

    C++中检查文件是否存在可通过std::ifstream或C++17的std::filesystem::exists实现;前者通过尝试打开文件并检查流状态,后者更简洁且跨平台。常见错误包括头文件缺失、路径错误、权限不足及编译器不支持C++17;跨平台时应使用std::filesystem::path…

    2025年12月19日
    000
  • c++中如何实现跨平台编译_c++跨平台编译方法

    答案是使用标准C++、CMake构建系统和条件编译实现跨平台编译。通过遵循标准语法、选用可移植库如std::filesystem和Boost.Asio、采用CMake生成各平台构建配置,并用预定义宏处理平台差异,结合CI自动化测试确保多平台兼容性。 在C++开发中,跨平台编译是指用同一份代码在不同操…

    2025年12月19日
    000
  • c++中如何生成固定长度的字符串_c++生成固定长度字符串方法

    使用构造函数可直接创建固定长度字符串,如std::string(10, ‘ ‘)生成10个空格;通过头文件结合字符集可生成指定长度的随机字符串;对于已有字符串,可通过截断或补全方式调整至固定长度,常用substr和append实现。 在C++中生成固定长度的字符串有多种方式,…

    2025年12月19日
    000
  • c++怎么使用protobuf_c++ Protobuf使用方法

    首先安装Protobuf编译器和库,然后编写.proto文件定义消息格式,接着使用protoc生成C++代码,再在程序中包含头文件并调用序列化与反序列化接口,最后编译时链接Protobuf库即可完成整个流程。 在C++中使用Protocol Buffers(简称Protobuf)需要经过几个步骤:定…

    2025年12月19日
    000
  • c++怎么实现读写锁_c++读写锁实现方法

    推荐使用C++17的std::shared_mutex实现读写锁,允许多个读线程共享访问、写线程独占访问;其通过std::shared_lock和std::unique_lock提供安全高效的并发控制,优于手动或Boost实现。 在C++中实现读写锁,核心目标是允许多个线程同时读取共享资源,但写操作…

    2025年12月19日
    000
  • c++中CMake怎么使用_CMake构建项目基本流程

    CMake构建流程为:编写CMakeLists.txt定义项目→创建build目录→运行cmake ..生成构建文件→执行cmake –build .编译→可选安装或测试,实现跨平台项目管理。 在C++项目中使用CMake构建系统,能有效管理编译流程、依赖关系和跨平台构建。下面介绍CMa…

    2025年12月19日
    000
  • C++如何获取当前时间_C++ 系统时间获取方法

    C++中获取系统时间主要有三种方法:1. 使用ctime库的time()和localtime()获取年月日时分秒;2. 通过strftime()自定义格式化时间字符串;3. 利用chrono库获取高精度时间或Unix时间戳,推荐现代C++项目使用chrono。 在C++中获取当前系统时间有多种方式,…

    2025年12月19日
    000
  • C++如何格式化输出_C++ 格式化输出方法

    C++中格式化输出主要有三种方法:①使用cout与,类型安全且灵活,适合C++风格开发;②采用printf来自,语法简洁高效,适用于熟悉C的场景;③利用stringstream进行复杂字符串拼接,便于构建格式化字符串。根据需求选择:追求安全性和可读性用cout,追求性能和简洁用printf,动态拼接…

    2025年12月19日
    000
  • c++中什么是RAII原则_c++ RAII原则解析

    RAII通过将资源管理绑定到对象生命周期来确保资源安全释放。在构造函数中获取资源,析构函数中释放,利用作用域自动调用析构,即使异常也能保证资源不泄漏。如FileHandler类在构造时打开文件,析构时关闭;标准库中unique_ptr、lock_guard等均体现此原则,实现内存、锁等资源的自动化管…

    2025年12月19日
    000
  • c++中的友元函数是什么_c++友元函数使用解析

    友元函数是C++中允许非成员函数访问类私有和保护成员的机制。它在类内用friend关键字声明,定义在类外,可直接访问类的所有成员,但不具有传递性和继承性,常用于运算符重载等场景。 友元函数是C++中一种特殊的机制,它允许一个非成员函数访问类的私有(private)和保护(protected)成员。正…

    2025年12月19日
    000
  • C++如何实现一个拷贝构造函数_C++ 拷贝构造函数实现方法

    拷贝构造函数用于用已存在对象初始化新对象,需实现深拷贝以避免浅拷贝问题;如MyArray类中,通过分配新内存并复制数据,确保每个对象独立管理动态数组,防止内存重复释放或数据污染。 拷贝构造函数是C++中用于用一个已存在的对象初始化新对象的特殊构造函数。正确实现拷贝构造函数,尤其是处理动态资源时,能避…

    2025年12月19日
    000
  • c++中标准输入输出流是什么_c++标准I/O流概念与操作

    C++标准输入输出流基于头文件,通过cin、cout、cerr和clog实现数据交互,使用>>和 在C++中,标准输入输出流(Standard I/O Streams)是用于程序与外部环境(通常是用户或终端)进行数据交换的核心机制。它基于头文件提供的类和对象,实现对输入和输出的面向对象式…

    2025年12月19日
    000
  • c++中的友元类是什么_c++友元类解析

    友元类是指通过friend关键字声明,使一个类能访问另一个类的私有和保护成员的机制。例如,class A声明class B为友元后,B可访问A的私有成员,但此关系单向、不可继承或传递。常用于高度耦合场景如容器与迭代器、设计模式协作或调试测试。使用时应避免滥用,优先考虑公有接口替代,以维护封装性与代码…

    2025年12月19日
    000
  • c++中C风格字符串和std::string怎么转换_c++ C风格字符串与string转换方法

    C风格字符串与std::string可相互转换:const char*可通过构造函数转为std::string;std::string通过c_str()获取C风格字符串指针,但需注意指针生命周期与只读限制。 在C++中,C风格字符串(即以空字符结尾的字符数组)和std::string是两种常见的字符…

    2025年12月19日
    000
  • c++中什么是POD(Plain Old Data)类型_c++ POD类型解析

    POD类型是C++中兼具平凡性和标准布局的类型,如int、float及无虚函数和访问控制的结构体,可用于与C兼容、静态初始化、memcpy操作等场景。 POD(Plain Old Data)类型是C++中一种特殊的数据类型,它指的是那些行为类似于C语言中的简单数据结构的类型。这类类型没有复杂的面向对…

    2025年12月19日
    000
  • c++ vector怎么查找特定元素_c++ vector查找元素方法

    使用std::find和std::find_if可在vector中查找元素,前者用于值匹配,后者支持条件查找,结合迭代器实现高效搜索。 在C++中,vector 是一个动态数组容器,常用于存储和管理数据。查找特定元素是常见的操作,可以通过标准库中的算法函数实现。最常用的方法是使用 std::find…

    2025年12月19日
    000
  • c++中如何使用递归实现树遍历_c++递归树遍历实现

    递归是实现二叉树前序、中序和后序遍历最直观的方法,通过定义TreeNode结构,分别在根节点处理前后调用递归函数实现三种遍历方式,注意空指针判断以确保程序安全。 在C++中,递归是实现树遍历最自然、最直观的方法。常见的树遍历方式包括前序、中序和后序遍历,它们都基于递归思想。下面以二叉树为例,说明如何…

    2025年12月19日
    000
  • c++怎么将所有字母转为大写或小写_字符串大小写转换方法

    c++kquote>使用std::transform配合::toupper或::tolower可实现字符串大小写转换,需包含和头文件,示例中将”C++ is FUN! 123″转为大写和小写,非字母字符保持不变,原地修改需先复制以保留原字符串。 在C++中,将字符串中的…

    2025年12月19日
    000
  • c++怎么使用namespace_C++命名空间的使用与最佳实践

    命名空间用于组织标识符防止冲突。使用namespace定义,如namespace Math { int add(int a, int b) { return a + b; } class Calculator { public: void show() { std::cout 在C++中,命名空间(…

    2025年12月19日
    000
  • c++中怎么写入文件_C++文件写入操作方法

    使用ofstream可实现C++文件写入,包含头文件后,通过ofstream创建文本或二进制文件,默认覆盖原内容,添加std::ios::app可追加写入,std::ios::binary用于二进制数据,需用reinterpret_cast转换指针类型,write()函数写入原始数据,操作后应检查i…

    2025年12月19日
    000

发表回复

登录后才能评论
关注微信