在HotSpot虚拟机中对象在内存中存储的布局可以分为3块区域:对象头(Header)、实例数据(Instance Data)和对其填充(Padding)
对象头
对象头包括三部分
-
第一部分Mark Word,用于存储对象自身的运行时数据,哈希码、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳。这部分数据在长度为32位和64位的虚拟机中分别为32bit和64bit。
-
类型指针,即对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例。并不是所有虚拟机实现都必须在对象数据上保留类型指针(查找对象的元数据信息并不一定要经过对象本身)。
-
如果对象是一个java数组,那在对象头中还必须有一块用于记录数组长度的数据,因为虚拟机可以通过普通Java对象的元数据信息确定Java对象的大小。但是从数组的元数据中却无法确定数组的大小。
实例数据
实例数据部分是对象真正存储的有效信息,也是程序代码中所定义的各种类型的字段内容,无论是从父类继承下来的,还是在子类中定义的,都需要记录起来。这部分存储顺序会受到虚拟机的分配策略参数(FieldsAllocationStyle)和字段在Java源码中定义顺序的影响。相同宽度的字段总是分配到一起,在满足这个的前提下,父类中定义的变量会出现在子类之前。如果CompactFields设置为true,那么子类之中较窄的变量也可能插入到父类变量的空隙中。
对其填充
并不是必然存在的,也没有特定的含义,它仅仅起着占位符的作用。由于HotSpot VM的自动内存管理系统要求对象的起始地址必须是8字节的整数倍。
实时内容请关注微信公众号,公众号与博客同时更新:程序员星星