引言:

        Java程序运行时无时无刻都有对象被创建出来,在语言层面上,创建对象只是通过一个new关键字,那么在JVM上对象是如何被创建出来的?

  1. JVM解析到new指令时,会去常量池中(方法区里面)定位该类的符号引用,并且检查该类是否被加载解析和    初始化。如果没有,执行类加载过程。
  2. 确认类加载后,jvm会为新生对象分配内存,该对象的内存在类加载完毕后就可以完全确定(文章末尾会说明为何)。分配内存可以有指针碰撞法和空闲列表法,并且需要处理同步的问题。
  3. 初始化对象内存空间(赋零值,每一个类型都有零值)。
  4.  对新生对象进行必要的设置,如对象是哪个类的实例,如何找到类的元数据信息,对象的哈希码,对象的Gc分代年龄等,这些信息存放在对象头中。

疑问:为什么类加载完毕后即可确定对象的大小?

         对象的存储布局可以分为:对象头,实例数据,对齐填充。

对于对象头,分为俩部分信息,第一部分存储对象自身的运行时数据,如哈希码,Gc分代年龄,锁状态    标志等等,这部分数据长度在32,64位机器上的长度分别为32,64。所以在特定的机器上,该部分长度是固定的。其实对象的头信息很多已经超出了32,64位,它被设计成可以根据自身状态来复用存储空间。

对象头还有一部分是类型指针,指向他的类元数据,表示该实例是哪个类的。这部分长度也是固定的,所以对象头的大小是固定的。

实例部分是对象真正存储的有效信息,也是在程序代码中定义的各种类型的字段内容,既然既定的类的字段是固定的,那么这部分肯定也是固定的并且是可以计算出来的。(除了字段之外,方法是怎么存在的?)

对齐填充不是必然存在的,由于jvm要求对象的起始地址必须为8字节的整数倍,所以当实例数据部分没有对齐时,需要通过对齐填充来补全,这部分也是可以通过计算直接计算出来的。

所以加载完类后可以直接确定该对象实例的大小。