sizeof计算数组长度仅对原生数组有效,传参后退化为指针会导致错误;std::size是C++17起更安全的统一替代方案,支持原生数组、std::array和标准容器。
sizeof 计算数组长度只对原生数组有效很多人写 sizeof(arr) / sizeof(arr[0]) 算元素个数,这确实能工作——但仅限于函数内部定义的原生数组(如 int arr[5];)。一旦数组退化为指针(比如传进函数参数),sizeof 就只返回指针大小(通常是 8 字节),结果完全错误。
常见错误现象:
- 函数参数写 void foo(int arr[]) 或 void foo(int* arr),里面用 sizeof(arr)/sizeof(*arr) → 永远得到 1(64 位下)或 2(32 位下)
- 用 std::vector 或 std::array 却误套这个公式 → 编译失败或结果无意义
int a[7]; size_t n = sizeof(a) / sizeof(a[0]); // ✅ 得到 7
void f(int x[]) { sizeof(x); } // ❌ 返回指针大小
std::size 是 C++17 起更安全的替代方案std::size 是标准库提供的非成员函数,专为容器和原生数组设计,底层对原生数组做模板推导,自动避开指针退化问题。它比手写 sizeof 表达式更健壮、可读性更好,且支持所有标准容器(std::vector、std::array、std::string 等)。
int a[10]; auto n = std::size(a); // 
✅ 推导出 10
std::array:std::array b; std::size(b); // ✅ 返回 3
std::vector:std::vector v(5); std::size(v); // ✅ 返回 5(等价于 v.size())
头文件(C++17 起).size() 而不是 std::size
对于标准容器(std::vector、std::string、std::deque 等),优先调用成员函数 .size() —— 它是 O(1) 的,语义明确,且不需要额外头文件。而 std::size 是统一接口,适合泛型代码中统一处理多种类型,但引入了间接层。
v.size() 更直接,IDE 补全友好,编译器更容易优化std::size(container) 可同时适配原生数组、std::array 和容器,避免特化std::size 对 C 风格字符串(const char*)无效,也不接受空指针;.size() 在容器为空时仍安全如果真需要在函数里拿到原生数组长度,唯一可靠方式是用数组引用作为参数——这样不会退化,sizeof 或 std::size 都能用。但这要求调用方传入的是确切大小的数组,灵活性差,实际项目中较少见。
template void f(int (&arr)[N]) { std::size(arr); // ✅ N 可推导 }
int*、std::vector 或不同长度的数组std::span(C++20)替代,兼顾安全与通用性原生数组长度计算本质是编译期行为,任何运行时企图“从指针反推长度”都不可靠。别信“我能从 malloc 地址里还原 size”,那不属于 C++ 语言保证的范畴。