java 中继承对象在堆内存中只创建一个连续对象实例,子类对象包含父类字段与自身字段,共享同一对象头,从而实现“is-a”关系的底层支持。
在 Java 中,当使用 new B(1, "b") 创建一个继承自 A 的子类 B 实例时,JVM 不会在堆中分别分配两个独立对象(一个 A、一个 B),而是仅分配一块连续的内存空间,用于存放整个 B 类型的对象。该对象的内存布局严格遵循继承链顺序,自上而下依次排布:
对象头(Object Header):包含 Mark Word(用于锁状态、GC 分代年龄等)和 Class Metadata Pointer(指向 B.class 的元数据),是 JVM 管理对象的统一入口;这种「扁平化、单块式」布局保证了类型兼容性:B 对象的起始地址(即对象头地址)可直接被视作 A 类型的引用——因为 A 的字段就位于该地址偏移量为 header_size 处,完全满足 A 的内存视图要求。这也是 B b = new B(...); A a = b; 能安全赋值的根本原因。
以下代码直观体现该机制:
public class A {
private int id;
public A(int id) { this.id = id; }
}
public class B extends A {
private String name;
public B(int id, String name) {
super(id);
this.name = name;
}
public static void main(String[] args) {
B b = new B(42, "Java");
// b 引用指向堆中唯一一块内存:[Header][id][name]
A aRef = b; // 合法向上转型 —— JVM 仅需确认该内存块兼容 A 的结构
System.out.println(aRef.getClass().getSimpleName()); // 输出: B
}
}⚠️ 注意事项:
简言之,Java 继承的本质不是“组合多个对象”,而是“扩展单个对象的字段视图”。理解这一内存模型,有助于深入掌握多态、强制转型、序列化字段顺序及内存对齐等高级主题。