在 C++ 中,空类(没有非静态成员变量)的大小通常为 1 字节 ,其根本目的是保证同一类型的两个不同对象在内存中有不同的地址。如果空类的大小为 0,那么连续存储的数组元素将全部重叠在同一地址上,这会破坏指针运算、对象标识等基本语义。
核心原理
- 编译器会隐式地向空类中插入一个 1 字节的占位符 (如
char dummy),使得每个对象都占用至少 1 字节的独立存储空间。 - 该占位符不会被初始化,也不影响访问权限,仅在计算对象大小时起作用。
伪代码演示
1. 如果允许大小为 0 会引发的问题
cpp
// 假设空类大小为 0(实际 C++ 不允许)
class Empty { /* 无成员 */ };
int main() {
Empty a, b; // 若 sizeof(Empty) == 0,则 &a == &b(地址冲突)
Empty arr[2]; // 若大小为 0,则 arr[0] 和 arr[1] 的地址相同
// 指针运算会出错:arr+1 == arr,无法区分不同元素
}
2. 编译器实际执行的"注入"逻辑(伪代码)
cpp
// 编译器内部处理:为空类添加一个 char 占位符
class Empty {
// 对于空类,编译器自动插入:
// char __placeholder__; // 占用 1 字节,不可见
};
// 等价于用户显式写出:
class Empty {
char dummy; // 使 sizeof(Empty) == 1
};
3. 对象标识对比(正常行为)
cpp
Empty a, b;
// &a != &b 因为每个对象至少占 1 字节
Empty arr[2];
// (void*)&arr[0] + 1 == (void*)&arr[1] // 地址连续且不同
注意事项
- 继承时可能发生"空基类优化"(EBO) :如果派生类没有其他成员,且基类是空类,编译器可能不分配基类的占位字节,从而节省空间。但这与"空类大小为 1"并不矛盾------那是针对独立对象或数组而言的。
- 虚继承会破坏 EBO:如虚继承自空类,编译器必须引入虚基类指针(8 字节),此时大小为 8 而非 1。
简单总结:空类大小为 1 的强制要求,本质上是为了满足 C++ 对象模型的唯一地址约束,通过一个隐藏的字节实现。