C++中的sizeof操作符返回什么?(对象所占内存字节数)

sizeof返回编译期确定的字节数,不依赖运行时状态;对数组、指针、字符串字面量、类对象(含空类、虚函数、对齐填充)等有特定规则,不能用于获取动态分配内存大小。

sizeof 返回的是编译期确定的字节数,不是运行时对象大小

sizeof 是一个编译期运算符,它不关心变量的实际值或运行时状态,只根据类型声明(或表达式类型)在编译时计算并替换为常量整数。即使对未定义、未初始化、甚至空指针解引用的表达式使用 sizeof,只要类型可推导,就不会报错。

例如:sizeof(*nullptr) 合法(返回所指类型的大小),sizeof(arr[1000]) 也合法——哪怕 arr 只有 5 个元素,因为 arr[1000] 的类型仍可推导(如 int)。

对类/结构体使用 sizeof 时,要考虑

对齐和填充

类对象的 sizeof 不等于所有成员大小之和。编译器会按目标平台的对齐要求插入填充字节(padding),以保证每个成员地址满足其对齐约束(如 double 通常需 8 字节对齐)。

  • 基类子对象、虚函数表指针(vptr)、虚基类偏移等都会计入 sizeof 结果
  • 空类(如 struct X {};)的 sizeof 为 1 —— 这是为了保证不同对象有唯一地址
  • 使用 #pragma pack(n)alignas 可改变默认对齐,从而影响 sizeof 结果

sizeof 对数组和指针的处理完全不同

这是最容易混淆的点:sizeof 能区分“数组类型”和“退化后的指针类型”,但仅限于作用域内具有完整数组类型的上下文。

  • int arr[10]; sizeof(arr) → 返回 40(假设 int 为 4 字节)
  • void func(int a[10]) { sizeof(a); } → 返回指针大小(如 8),因为形参 a 实际是 int*
  • sizeof("hello") → 返回 6,包含末尾的 \0;而 sizeof(char*) 是指针大小

不能用 sizeof 获取动态分配内存的大小

sizeofnew 出来的指针、malloc 返回的地址、或容器(如 std::vector)的内部数据指针,一律只返回指针本身大小,而非其所指内存块长度。

例如:

int* p = new int[100];
std::cout << sizeof(p) << "\n";     // 输出 8(64 位系统下指针大小)
std::cout << sizeof(*p) << "\n";   // 输出 4(int 大小)
// 没有办法仅凭 p 得到 100 这个数量

真正需要运行时大小信息时,必须额外保存(如用 std::vector::size()、或自己记一个 count 变量)。依赖 sizeof 去“探测”堆上对象尺寸,注定失败。