一、简介
在HotSpot虚拟机里,对象在堆内存中的存储布局可以划分为三个部分:对象头(Header)、实例数据(Instance Data)和对齐填充(Padding)。其中对其填充是为了保证对象是8字节的倍数!
其中对象头可以再次进行细分,分为对象标记(Mark Word)
和类元信息(类型指针)
,对于数组对象来说对象头又会多一个Length
部分。在64位系统中,Mark Word占了8个字节,类型指针占了8个字节,一共是16个字节。
从openJDK的oop.hpp
源码中可以看到:_mark字段是mark word,_metadata是类指针klass pointer,对象头(object header)即是由这两个字段组成.
二、对象头之MarkWord
可以发现哈希码、GC标记,GC次数,同步锁标记,偏向锁持有者都在Mark Word
里面.
这些信息都是与对象自身定义无关的数据,所以MarKWord被设计成一个非固定的数据结构以便在极小的空间内存存储尽量多的数据。它会根据对象的状态复角首己的存储空间,也就是说在运行期间MarkWord里存储的数据会随着锁标志做的变化而变化。
下图是markOop.hpp
的源码
通过上图,我们一般会归纳总结出下面的这张图片:
由于对象的分代年龄只占4bit,所以对象最大年龄是二进制的1111
,15岁!当给jvm设置最大年龄为16时,运行就会报错!
三、对象头之类型指针
从下图可以看出堆空间对象的类型指针,指向了方法区的类元信息。
四、实例数据和Padding
1.实例数据:存放类的属性(Field)数据信息,包括父类的属性信息
2.对齐填充:虚拟机要求对象起始地址必须是8字节的整数倍。填充数据不是必须存在的,仅仅是为了字节对齐这部分内存按8字节补充对齐。
五、JOL工具分析对象内存布局
<dependency>
<groupId>org.openjdk.jol</groupId>
<artifactId>jol-core</artifactId>
<version>0.10</version>
</dependency>
通过下面的方法可以查看当前JDK的设置情况!
下面是打印出对象的内存布局,可以看出new Object()占16字节(8字节Mark word+4字节类型指针+4字节Padding)
//下面两种情况打印的结果是一样的
System.out.println(ClassLayout.parseInstance(new Object()).toPrintable());
//或
class M{
}
System.out.println(ClassLayout.parseInstance(new M()).toPrintable());
但如果类中定义了字段,比如说下面的情况,我们就发现内存布局中多出了字段带来的实例数据。
class M{
int id;
boolean flag;
}
System.out.println(ClassLayout.parseInstance(new M()).toPrintable());
六、压缩指针
我们前面说了,对于64为的hotspot虚拟机,markword和classpoint都占8字节,但通过上面的JOL工具我们发现classpoint只有4字节,结果与我们预测的不一致的原因是,jvm默认进行了压缩指针的操作,可以通过给jvm加上下面的参数查看:
-XX:+PrintCommandLineFlags -version
我们可以通过下面的jvm参数来关闭压缩指针,发现类型指针占8字节了!对象头就有16字节了所以无需padding
-XX:-UseCompressedClassPointers