在HotSpot虚拟机中,对象在内存中存储的布局可以分为3块区域:
- 对象头(Header)
- 实例数据(Instance Data)
- 对齐填充(Padding)
对象头
HotSpot虚拟机的对象头包括两部分信息:
- 运行时数据:用于存储对象自身的运行时数据,如HashCode、GC分代年龄、锁状态标志、线程持有的锁、 偏向线程ID、偏向时间戳等
- 类型指针: 即对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例;当然,查找对象的元数据信息并不一定要通过对象本身,并不是所有的虚拟机实现都必须在对象数据上保留类型指针
如果对象是Java数组,那在对象头中还必须有一块用于标记数组长度的数据,因为虚拟机可以通过普通对象的
元数据确定对象的大小,但是从数组的元数据中却无法确定数组的大小。
实例数据
实例数据部分是对象真正存储的有效信息,也是在程序代码中所定义的各种类型的字段内容。父类和子类中定义的会被记录下来,存储顺序受虚拟机分配策略和字段在Java源码中定义的顺序影响。
HotSpot虚拟机默认的分配策略为 longs/doubles, ints, shorts/chars, bytes/booleans, oops(
Ordinary Object Pointers),从分配策略上可以看出,相同宽度的字段总是被分配到一起。
对齐填充
它起着占位符的作用。因为HotSpot VM的自动内存管理系统要求对象的起始地址必须是8字节的整数倍,对象头部正好是8字节的整数倍,因此,当对象实例数据部分没有对齐时,就需要通过对齐填充来补全。