①. 对象在堆内存中的存储布局
-
①. 对象内部结构分为:对象头、实例数据、对齐填充(保证8个字节的倍数)。
-
②. 对象头分为对象标记(markOop)和类元信息(klassOop),类元信息存储的是指向该对象类元数据(klass)的首地址。
②. 对象头(Header)
①. 对象标记Mark Word
①. 默认存储 (哈希值(HashCode )、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳)等信息
- 这些信息都是与对象自身定义无关的数据,所以MarkWord被设计成一个非固定的数据结构以便在极小的空间内存存储尽量多的数据。
- 它会根据对象的状态复用自己的存储空间,也就是说在运行期间MarkWord里存储的数据会随着锁标志位的变化而变化。
②. 在64位系统中,Mark Word占了8个字节,类型指针占了8个字节,一共是16个字节
②. 类元信息(又叫类型指针)
- 对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例
③. 实例数据
- 说明:它是对象真正存储的有效信息,包括程序代码中定义的各种类型的字段(包括从父类继承下来的和本身拥有的字段) 规则:
- 相同宽度的字段总被分配在一起
- 父类中定义的变量会出现在子类之前
- 如果CompactFields参数为true(默认为true),子类的窄变量可能插入到父类变量的空隙
④. 对齐填充
-
①. 虚拟机要求对象起始地址必须是8字节的整数倍。填充数据不是必须存在的,仅仅是为了字节对齐。这部分内存按8字节补充对齐。
-
②. 不是必须的,也没特别含义,仅仅起到占位符作用
-
③. 解释如下图:
⑤. 总结
⑥. 再说对象头的MarkWord
markword(64位)分布图,对象布局、GC回收和后面的锁升级就是:对象标记MarkWord里面标志位的变化
markOop.hpp
hash: 保存对象的哈希码
age: 保存对象的分代年龄
biased_lock: 偏向锁标识位
lock: 锁状态标识位
JavaThread* :保存持有偏向锁的线程ID
epoch: 保存偏向时间戳
⑦. 聊聊Object obj = new Object()
<!--
JAVA object layout
官网:http://openjdk.java.net/projects/code-tools/jol/
定位:分析对象在JVM的大小和分布
-->
<dependency>
<groupId>org.openjdk.jol</groupId>
<artifactId>jol-core</artifactId>
<version>0.9</version>
</dependency>
public class JOLDemo{
public static void main(String[] args){
Object o = new Object();
System.out.println( ClassLayout.parseInstance(o).toPrintable());
}
}
结果呈现说明
GC年龄采用4位bit存储,最大为15,例如MaxTenuringThreshold参数默认值就是15
-XX:MaxTenuringThreshold=16
因为GC年龄占4位,最大就是1111=15
手动关闭压缩再看看 -XX:-UseCompressedClassPointers
说到这里,如果Object是个空对象,那么new Object 就是占16个字节