虚拟机遇到一条new指令时,首先将去检查这个指令的参数是否能在常量池中定位到这个类的符号引用,并且检查这个符号引用代表的类是否已经被加载过、解析和初始化过。如果没有,那必须先执行响应的类加载过程。
在类加载检查通过后,接下来虚拟机将为新生对象分配内存。对象所需的内存大小在类加载完成后便可确定,为对象分配空间的任务等同于吧一块确定大小的内存从Java堆中划分出来。内存分配的查找方式有“指针碰撞”和“空闲列表”两种。
选择以上两种方式中的哪一种,取决于Java堆内存是否规整。而Java堆内存是否规整取决于GC收集器的算法是“标记-清除”,还是“标记=整理”。
内存分配完成后,虚拟机需要将分配到的内存空间都初始化为零值(不包括对象头),这一步操作保证了对象的实例字段在Java代码中可以不赋初始值就直接使用,程序能访问到这些字段的数据类型所对应的零值。
初始化零值完成后,虚拟机要对新对象进行必要的设置,例如这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希码、对象的GC分代年龄等信息。这些信息存放在对象头中。另外,根据虚拟机当前运行的状态不同,如是否启用偏向锁等,对象头会有不同的设置方式。
在上面工作都完成后,从虚拟机的视角来看,一个新的对象已经产生了,但从Java程序的视角来看,对象创建才开始,<init>
构造方法还没有执行,目前所有的字段都还为零。所以一般来说,执行new指令之后会接着执行<init>
构造方法,把对象按照逻辑的意愿进行初始化,这样一个真正可用的对象才算完整创建出来。