0

0

如何在C++中获取数组的长度_C++数组长度计算方法

冰火之心

冰火之心

发布时间:2025-10-10 11:32:01

|

935人浏览过

|

来源于php中文网

原创

获取数组长度取决于数组类型:C风格静态数组可用sizeof(arr)/sizeof(arr[0]),动态数组需手动记录长度,现代C++推荐使用std::vector和std::array的size()方法,C++17起可统一用std::size()获取各类容器和数组的长度。

如何在c++中获取数组的长度_c++数组长度计算方法

在C++中获取数组的长度,最直接的答案是:这取决于你讨论的是哪种“数组”。对于编译时大小固定的C风格数组,你可以用sizeof(array) / sizeof(array[0])来计算元素个数。但如果你处理的是指向动态分配内存的指针,或者将数组作为函数参数传递后,这种方法就行不通了,因为此时你面对的只是一个指针,而非完整的数组结构。现代C++中,std::vectorstd::array提供了更安全、更直接的size()方法。

解决方案

在C++中,获取数组长度的方法多样,但其适用场景和背后原理各有不同。理解这些差异,是避免常见编程陷阱的关键。

1. C风格静态数组(编译时已知大小)

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

这是最经典的场景。当你声明一个像 int arr[10]; 这样的数组时,编译器在编译时就知道它的大小。此时,你可以使用 sizeof 运算符来获取其长度:

#include 

int main() {
    int static_arr[] = {1, 2, 3, 4, 5}; // 编译器自动推断大小
    // 或者 int static_arr[5] = {1, 2, 3, 4, 5};

    // sizeof(static_arr) 获取整个数组占用的字节数
    // sizeof(static_arr[0]) 获取单个元素占用的字节数
    // 两者相除即为元素个数
    size_t length = sizeof(static_arr) / sizeof(static_arr[0]);
    std::cout << "静态数组的长度是: " << length << std::endl; // 输出 5

    return 0;
}

这里 sizeof(static_arr) 返回的是整个数组在内存中占据的总字节数,而 sizeof(static_arr[0]) 返回的是数组中第一个元素(也是任意一个元素)所占据的字节数。两者相除,自然就是元素的个数。这个方法简单直接,但它有个致命的限制:只对完整的数组类型有效。

2. 指针(动态数组或数组作为函数参数)

当你通过 new 运算符动态分配内存,或者将C风格数组作为函数参数传递时,情况就完全不同了。在这些场景下,你手上持有的不再是一个“数组”,而仅仅是一个指向数组首元素的指针

#include 

void processArray(int* arr_ptr, size_t size) { // 接收指针和大小
    // 在这里,sizeof(arr_ptr) 只会返回指针本身的大小 (通常是4或8字节)
    // 而不是它所指向的数组的大小
    std::cout << "在函数内部,指针的大小是: " << sizeof(arr_ptr) << " 字节" << std::endl;
    std::cout << "我们必须依赖传入的 size 参数: " << size << std::endl;
}

int main() {
    int* dynamic_arr = new int[10]; // 动态分配10个int的数组
    // ... 对 dynamic_arr 进行操作 ...

    // 错误示范:这里 sizeof(dynamic_arr) 得到的是指针的大小,不是数组大小
    // size_t length_fail = sizeof(dynamic_arr) / sizeof(dynamic_arr[0]); // 结果会是1或2,而不是10
    // 动态数组的长度必须由程序员自己管理和记住
    size_t dynamic_arr_len = 10;
    std::cout << "动态数组的长度是: " << dynamic_arr_len << std::endl;

    processArray(dynamic_arr, dynamic_arr_len);

    delete[] dynamic_arr; // 释放内存
    return 0;
}

processArray 函数中,arr_ptr 仅仅是一个 int* 类型的指针,它“忘掉”了它曾经是一个数组的身份。这就是所谓的“数组到指针的衰退 (array-to-pointer decay)”。因此,如果你不额外传递数组的长度信息,函数内部是无法得知其真实长度的。对于动态分配的数组,你必须在分配时就记住其长度,并在需要时传递这个长度。

3. 现代C++容器 (std::vectorstd::array)

现代C++提供了更安全、更方便的容器来替代C风格数组,它们内部都封装了长度信息。

  • std::vector (动态大小数组)

    std::vector 是一个动态大小的序列容器,它可以根据需要自动增长或收缩。它提供了 size() 成员函数来获取当前元素的个数。

    #include 
    #include 
    
    int main() {
        std::vector myVector = {10, 20, 30, 40};
        myVector.push_back(50); // 运行时增加元素
    
        size_t length = myVector.size(); // 直接获取长度
        std::cout << "std::vector 的长度是: " << length << std::endl; // 输出 5
    
        return 0;
    }

    std::vector 几乎总是处理动态数组的首选。

  • std::array (固定大小数组的容器封装)

    std::array 是C++11引入的,它是一个固定大小的数组,但提供了C++容器的接口,包括 size() 方法。它在上分配内存,性能与C风格数组相当,但提供了更好的类型安全和接口。

    #include 
    #include 
    
    int main() {
        std::array myArray = {1.1, 2.2, 3.3};
    
        size_t length = myArray.size(); // 直接获取长度
        std::cout << "std::array 的长度是: " << length << std::endl; // 输出 3
    
        return 0;
    }

    对于编译时大小已知的固定数组,std::array 是一个非常好的替代品。

为什么sizeof在某些情况下无法准确获取C++数组的长度?

这其实是C++语言设计中一个非常核心且容易让人困惑的特性,我们称之为“数组到指针的衰退”(Array-to-pointer decay)。简单来说,当一个C风格数组在大多数表达式中使用时,它会自动“衰退”或隐式转换为指向其第一个元素的指针。

Convai Technologies Inc.
Convai Technologies Inc.

对话式 AI API,用于设计游戏和支持端到端的语音交互

下载

想象一下,你有一个盒子,上面写着“里面有5个苹果”。sizeof运算符在应用于这个盒子本身(也就是完整的数组变量)时,能准确地告诉你这个盒子有多大,进而推算出里面有多少个苹果。但当你把这个盒子交给别人,而别人只知道你给了他一个“苹果的地址”(一个指针),他并不知道这个地址后面跟着多少个苹果。他只知道他手里的这个地址标签本身有多大(比如4字节或8字节)。

具体来说:

  1. sizeof应用于完整数组类型时: 当 sizeof 直接作用于一个在当前作用域内完整定义的C风格数组变量时,例如 int arr[10];sizeof(arr) 会返回整个数组所占用的总字节数。因为编译器在编译时知道 arr 是一个包含10个 int 元素的数组,每个 int 占4字节(假设),那么 sizeof(arr) 就会是 10 * 4 = 40 字节。这时,sizeof(arr) / sizeof(arr[0]) 就能正确计算出元素数量。

  2. sizeof应用于指针类型时: 当一个数组被用作函数参数,或者通过 new 动态分配后,你得到的是一个 int* 这样的指针。此时,sizeof 作用于这个指针变量,例如 int* ptr;sizeof(ptr) 返回的只是指针变量本身在内存中占用的字节数(在32位系统上通常是4字节,在64位系统上通常是8字节),而不是它所指向的内存块的大小。 例如:

    void func(int arr_param[]) { // 实际上 arr_param 是 int* 类型
        // sizeof(arr_param) 得到的是指针的大小,不是数组大小
        // sizeof(arr_param[0]) 得到的是 int 的大小
        // 结果会是 (4或8) / 4 = 1 或 2,而不是实际的数组长度
        std::cout << "func内部计算的长度: " << sizeof(arr_param) / sizeof(arr_param[0]) << std::endl;
    }
    
    int main() {
        int myArr[5] = {1,2,3,4,5};
        func(myArr); // 数组 myArr 衰退为 int* 传递给 func
        return 0;
    }

    运行上述代码,你会发现 func 内部计算出的长度是错误的。这就是 sizeof 在指针上无法获取数组长度的根本原因。编译器在处理 func(myArr) 时,会将 myArr 的地址传递给 funcfunc 接收到的只是一个地址,它不再知道这个地址后面跟着多少个 int

这种衰退是C语言(以及C++继承C的部分)的一个特性,它使得函数可以接受不同大小的数组作为参数(因为它们都衰退为同一种指针类型),但也带来了长度信息丢失的问题。因此,当处理C风格数组时,如果你需要知道其长度,要么确保它是一个完整的数组变量,要么就得显式地传递长度信息。

在现代C++中,管理数组长度有哪些推荐的最佳实践?

现代C++编程哲学强调类型安全、资源管理自动化和避免C风格的陷阱。因此,对于数组长度的管理,我们有了一套更健壮、更清晰的最佳实践:

  1. 首选 std::vector 处理动态大小序列: 几乎所有需要动态调整大小的“数组”场景,都应该优先考虑 std::vector。它不仅提供了 size() 方法来获取当前元素数量,还负责内存的自动管理(分配和释放),极大减少了内存泄漏和越界访问的风险。它的接口丰富,支持各种算法和迭代器操作。

    #include 
    #include 
    
    std::vector getData() {
        std::vector data = {1, 2, 3};
        data.push_back(4);
        return data;
    }
    
    int main() {
        std::vector numbers = getData();
        std::cout << "Vector length: " << numbers.size() << std::endl; // 始终正确
        // 使用 range-based for 循环,无需关心长度
        for (int n : numbers) {
            std::cout << n << " ";
        }
        std::cout << std::endl;
        return 0;
    }

    使用 std::vector 可以让你摆脱手动管理内存和传递长度的烦恼,代码更简洁、更安全。

  2. 使用 std::array 处理固定大小数组: 如果数组的大小在编译时就已知且固定,std::array 是C风格静态数组的完美替代品。它提供了 size() 方法,并且像 std::vector 一样,支持迭代器和标准算法,但其内存是在栈上分配的,性能与C风格数组相同,且避免了数组衰退的问题。

    #include 
    #include 
    
    void processFixedArray(std::array& arr) { // 传递引用,避免拷贝
        std::cout << "Fixed array length in function: " << arr.size() << std::endl;
        // ...
    }
    
    int main() {
        std::array scores = {90.5, 88.0, 92.5, 78.0, 95.0};
        std::cout << "std::array length: " << scores.size() << std::endl;
        processFixedArray(scores);
        return 0;
    }

    std::array 结合了C风格数组的效率和C++容器的安全性与便利性。

  3. 为C风格数组/指针显式传递长度: 在极少数情况下,你可能不得不使用C风格的动态分配数组(new[])或与C库交互。在这种情况下,必须将数组的长度作为单独的参数传递给任何处理该数组的函数。这是避免越界访问的唯一可靠方法。

    #include 
    
    void printCArray(const int* arr, size_t len) { // 必须传入长度
        for (size_t i = 0; i < len; ++i) {
            std::cout << arr[i] << " ";
        }
        std::cout << std::endl;
    }
    
    int main() {
        size_t count = 7;
        int* rawData = new int[count];
        for (size_t i = 0; i < count; ++i) {
            rawData[i] = i * 10;
        }
    
        printCArray(rawData, count); // 显式传递长度
    
        delete[] rawData;
        return 0;
    }

    这种方式虽然有效,但需要程序员手动管理长度,容易出错,因此应尽量避免。

  4. 使用迭代器和范围 (Range-based for loops, Algorithms): 现代C++鼓励使用迭代器和基于范围的(range-based)算法。如果你有一个容器(如 std::vector, std::array)或任何支持迭代器的序列,很多时候你甚至不需要显式获取其长度。例如,范围 for 循环会自动遍历所有元素。

    #include 
    #include  // For std::accumulate
    #include 
    
    int main() {
        std::vector data = {1, 2, 3, 4, 5};
    
        // Range-based for loop: 不需要知道长度
        for (int x : data) {
            std::cout << x << " ";
        }
        std::cout << std::endl;
    
        // 使用标准算法:通常也通过迭代器范围工作
        long long sum = std::accumulate(data.begin(), data.end(), 0LL);
        std::cout << "Sum: " << sum << std::endl;
    
        return 0;
    }

    这种方式将长度管理的责任从程序员转移到容器和算法,使得代码更具通用性和安全性。

总结来说,在现代C++中,尽量使用 std::vectorstd::array,它们提供了内置的 size() 方法,并且通过类型系统和RAII(资源获取即初始化)原则,极大地简化了数组长度的管理和内存安全。只有在不得不与C风格代码交互时,才考虑手动传递长度参数。

C++17及更高版本中,有没有更现代、更通用的获取数组长度的方法?

当然有!C++17引入了 std::size 函数模板,它提供了一种统一且类型安全的方式来获取各种容器和C风格数组的长度。这是一个非常棒的改进,因为它消除了在使用不同类型数组时记住不同获取长度方法的需要。

std::size 的用法和优势:

std::size 是一个非成员函数模板,它可以作用于:

  • C风格数组:像 sizeof(arr) / sizeof(arr[0]) 一样工作,但语法更简洁。
  • std::array:调用其 size() 成员函数。
  • std::vectorstd::stringstd::list 等标准容器:调用其 size() 成员函数。

这样,无论你处理的是哪种“数组”,都可以用 std::size 来获取其长度,代码的一致性大大提高。

#include 
#include 
#include 
#include 
#include  // 包含 std::size

int main() {
    // 1. C风格静态数组
    int c_arr[] = {10, 20, 30, 40, 50};
    std::cout << "C风格数组的长度: " << std::size(c_arr) << std::endl; // 输出 5

    // 2. std::vector
    std::vector myVector = {1.1, 2.2, 3.3};
    myVector.push_back(4.4);
    std::cout << "std::vector 的长度: " << std::size(myVector) << std::endl; // 输出 4

    // 3. std::array
    std::array myArray = {'H', 'e', 'l', 'l', 'o', '\0'};
    std::cout << "std::array 的长度: " << std::size(myArray) << std::endl; // 输出 6

    // 4. std::string (虽然不是数组,但也是序列容器)
    std::string myString = "World";
    std::cout << "std::string 的长度: " << std::size(myString) << std::endl; // 输出 5

    // 注意:std::size 不能用于指向动态分配内存的裸指针
    // int* dynamic_ptr = new int[10];
    // std::cout << std::size(dynamic_ptr) << std::endl; // 编译错误!
    // delete[] dynamic_ptr;

    return 0;
}

std::size 的优势总结:

  • 统一接口:无论底层是C风格数组还是标准容器,都使用 std::size(obj) 这一种语法。这使得代码更易读、更易维护。
  • 类型安全:它是一个函数模板,会在编译时检查类型。如果尝试对一个裸指针使用 std::size,编译器会报错,从而避免了 sizeof(ptr) / sizeof(ptr[0]) 这种错误计算指针大小的陷阱。
  • 减少错误:通过提供统一且安全的接口,它减少了程序员因混淆不同类型数组的长度获取方法而导致的错误。

除了 std::size,C++17还引入了 std::emptystd::data,它们与 std::size 一起,为处理各种序列数据提供了更现代、更一致的接口。这些函数模板都位于 头文件中。

所以,如果你在C++17或更高版本中编程,强烈推荐使用 std::size 来获取数组或容器的长度,它无疑是目前最现代、最通用、最安全的方法之一。这体现了现代C++追求一致性、安全性和易用性的设计理念。

相关专题

更多
C语言变量命名
C语言变量命名

c语言变量名规则是:1、变量名以英文字母开头;2、变量名中的字母是区分大小写的;3、变量名不能是关键字;4、变量名中不能包含空格、标点符号和类型说明符。php中文网还提供c语言变量的相关下载、相关课程等内容,供大家免费下载使用。

387

2023.06.20

c语言入门自学零基础
c语言入门自学零基础

C语言是当代人学习及生活中的必备基础知识,应用十分广泛,本专题为大家c语言入门自学零基础的相关文章,以及相关课程,感兴趣的朋友千万不要错过了。

611

2023.07.25

c语言运算符的优先级顺序
c语言运算符的优先级顺序

c语言运算符的优先级顺序是括号运算符 > 一元运算符 > 算术运算符 > 移位运算符 > 关系运算符 > 位运算符 > 逻辑运算符 > 赋值运算符 > 逗号运算符。本专题为大家提供c语言运算符相关的各种文章、以及下载和课程。

351

2023.08.02

c语言数据结构
c语言数据结构

数据结构是指将数据按照一定的方式组织和存储的方法。它是计算机科学中的重要概念,用来描述和解决实际问题中的数据组织和处理问题。数据结构可以分为线性结构和非线性结构。线性结构包括数组、链表、堆栈和队列等,而非线性结构包括树和图等。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

256

2023.08.09

c语言random函数用法
c语言random函数用法

c语言random函数用法:1、random.random,随机生成(0,1)之间的浮点数;2、random.randint,随机生成在范围之内的整数,两个参数分别表示上限和下限;3、random.randrange,在指定范围内,按指定基数递增的集合中获得一个随机数;4、random.choice,从序列中随机抽选一个数;5、random.shuffle,随机排序。

597

2023.09.05

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

523

2023.09.20

c语言get函数的用法
c语言get函数的用法

get函数是一个用于从输入流中获取字符的函数。可以从键盘、文件或其他输入设备中读取字符,并将其存储在指定的变量中。本文介绍了get函数的用法以及一些相关的注意事项。希望这篇文章能够帮助你更好地理解和使用get函数 。

639

2023.09.20

c数组初始化的方法
c数组初始化的方法

c语言数组初始化的方法有直接赋值法、不完全初始化法、省略数组长度法和二维数组初始化法。详细介绍:1、直接赋值法,这种方法可以直接将数组的值进行初始化;2、不完全初始化法,。这种方法可以在一定程度上节省内存空间;3、省略数组长度法,这种方法可以让编译器自动计算数组的长度;4、二维数组初始化法等等。

599

2023.09.22

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

61

2026.01.14

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Django 教程
Django 教程

共28课时 | 3.1万人学习

Excel 教程
Excel 教程

共162课时 | 11.8万人学习

MongoDB 教程
MongoDB 教程

共17课时 | 2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号