c++ sizeof和strlen区别 c++内存大小计算详解

sizeof是编译时运算符,返回类型或对象所占字节数,对数组返回总大小(含'\0'),对指针只返回指针大小;2. strlen是运行时函数,计算以'\0'结尾的字符串中有效字符数,不包含'\0',需逐字节扫描至结束符。

sizeofstrlen 看似都和“大小”有关,但它们的作用、计算方式和使用场景完全不同。理解它们的区别对掌握 C++ 内存管理至关重要。

1. sizeof:编译时运算符,计算类型或对象所占字节数

sizeof 是一个编译时运算符,返回的是数据类型或变量在内存中占用的字节数(size_t 类型)。它不关心内容,只关心类型大小。

关键点:

  • 对数组:返回整个数组的字节大小,包括末尾的 '\0'(如果存在)
  • 对指针:只返回指针本身的大小(如 8 字节 on 64-bit)
  • 对字符串字面量:包含结尾的 '\0',所以 "hello" 是 6 字节
  • 在编译阶段确定结果,不执行程序也能算出

示例:

char arr[] = "hello";        // 实际是 char[6]
char* ptr = arr;

sizeof(arr); // 6(6个字符,含'\0') sizeof(ptr); // 8(64位系统下指针大小) sizeof("hello"); // 6(字符串字面量也占6字节) sizeof(int); // 通常为4

2. strlen:运行时函数,计算字符串实际字符数

strlen 是 中的函数,用于计算以 '\0' 结尾的字符串中**有效字符的数量**,不包含 '\0' 本身。

关键点:

  • 只适用于以 '\0' 结尾的字符数组或字符串指针
  • 从起始地址开始逐字节扫描,直到遇到 '\0' 停止
  • 返回值是 size_t,表示字符个数
  • 在运行时计算,依赖实际内存内容

示例:

char arr[] = "hello";
char* ptr = "world";

strlen(arr); // 5 strlen(ptr); // 5 strlen(""); // 0

3. 常见误区与对比

最容易混淆的情况出现在指针和数组传参时:

void func(char arr[]) {
    cout << sizeof(arr) << endl; // 输出指针大小(如8),不是数组原始大小!
    cout << strlen(arr) << endl; // 正确输出字符串长度(如5)
}

因为形参 arr 实际上是指针,sizeof 拿不到原始数组长度。

另一个典型例子:

char* str = "hello";
sizeof(str);  // 8(指针大小)
strlen(str);  // 5(字符串内容长度)

4. 内存大小计算实用建议

如何正确评估内存使用?记住以下原则:

  • 定义字符数组存放字符串时,确保空间足够容纳 '\0'
  • 用 sizeof 判断数组是否能放下某个字符串:sizeof(arr) > strlen(str) + 1
  • 动态分配内存时,根据 strlen + 1 来 malloc 或 new
  • 结构体大小受内存对齐影响,sizeof(struct) 不一定等于成员之和

例如:

char buffer[10];
strcpy(buffer, "hi");           // 安全,strlen("hi")=2,需要3字节
// sizeof(buffer)=10 >= 3

基本上就这些。sizeof 关注“占了多少内存”,strlen 关心“字符串有多长”。搞清用途,避免混用,才能写出安全高效的 C++ 代码。