
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 管理对象的统一入口;
- 父类字段(Inherited Fields):按声明顺序填充 A 类的实例字段(如 private int id);
- 子类字段(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
}
}⚠️ 注意事项:
本文档主要讲述的是android界面布局详解;在通过“Hello World!”介绍Android中的布局问题之前,不得不先介绍一下Android中的用户界面,因为布局问题也是用户界面问题之一。在一个Android应用程序中,用户界面通过View和ViewGroup对象构建。Android中有很多种Views和ViewGroups,他们都继承自View类。View对象是Android平台上表示用户界面的基本单元。希望本文档会给有需要的朋友带来帮助;感兴趣的朋友可以过
立即学习“Java免费学习笔记(深入)”;
- 字段排列可能受 JVM 实现(如 HotSpot)的字段重排序优化影响(例如将引用类型字段对齐到 8 字节边界),但逻辑顺序(父类字段在前、子类字段在后)和内存连续性始终不变;
- static 字段和方法不参与对象实例内存布局,它们属于类元数据,存储在方法区(JDK 8+ 为 Metaspace);
- 构造器调用链(super(...))仅负责初始化对应字段,并不触发额外对象分配;
- 多层继承(如 C extends B extends A)会线性扩展该结构:[Header][A-fields][B-fields][C-fields]。
简言之,Java 继承的本质不是“组合多个对象”,而是“扩展单个对象的字段视图”。理解这一内存模型,有助于深入掌握多态、强制转型、序列化字段顺序及内存对齐等高级主题。









