Java 中继承对象的内存布局详解

java 中继承对象在堆内存中只创建一个连续对象实例,子类对象包含父类字段与自身字段,共享同一对象头,从而实现“is-a”关系的底层支持。

在 Java 中,当使用 new B(1, "b") 创建一个继承自 A 的子类 B 实例时,JVM 不会在堆中分别分配两个独立对象(一个 A、一个 B),而是仅分配一块连续的内存空间,用于存放整个 B 类型的对象。该对象的内存布局严格遵循继承链顺序,自上而下依次排布:

  1. 对象头(Object Header):包含 Mark Word(用于锁状态、GC 分代年龄等)和 Class Metadata Pointer(指向 B.class 的元数据),是 JVM 管理对象的统一入口;
  2. 父类字段(Inherited Fields):按声明顺序填充 A 类的实例字段(如 private int id);
  3. 子类字段(Own Fields):紧随其后存放 B 类自身声明的实例字段(如 private String name)。

这种「扁平化、单块式」布局保证了类型兼容性: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
 

} }

⚠️ 注意事项:

  • 字段排列可能受 JVM 实现(如 HotSpot)的字段重排序优化影响(例如将引用类型字段对齐到 8 字节边界),但逻辑顺序(父类字段在前、子类字段在后)和内存连续性始终不变;
  • static 字段和方法不参与对象实例内存布局,它们属于类元数据,存储在方法区(JDK 8+ 为 Metaspace);
  • 构造器调用链(super(...))仅负责初始化对应字段,并不触发额外对象分配;
  • 多层继承(如 C extends B extends A)会线性扩展该结构:[Header][A-fields][B-fields][C-fields]。

简言之,Java 继承的本质不是“组合多个对象”,而是“扩展单个对象的字段视图”。理解这一内存模型,有助于深入掌握多态、强制转型、序列化字段顺序及内存对齐等高级主题。