在虚拟机遇到一条new的指令时先會在常量池中定位这个类的符号引用,并且检查这个类是否被加载如果没有,那么必须先执行加载过程在这个类加载完成后,接下来虛拟机将为新生对象分配内存对象所需的内存在类加载完成后就会完全确定,也就是从堆中划分出一个区域如果堆中的内存是绝对完整的,那么采用指针碰撞的方式即可完成内存的分配也就是移动指针的位置即可,如果堆的内存不是绝对完整的那么虚拟机就必须维護一个表来记录空闲的区域,这种方法叫做空闲列表 采用哪种方法由java堆是否完整决定而java堆是否完整由采用哪种垃圾收集算法决定,比如Serial、ParNew等带ComPact(整理)过程的收集器采用指针碰撞,采用CMS这种基于Mark-Sweep算法的收集器时采用空闲列表方式。
还有一个问题需要考虑那就是在分配空間的时候即使是移动指针这种方式,也不是线程安全的也可能产生线程安全问题,因此要解决这个问题一种是对分配内存空间的动作進行同步操作,另一种是每个线程分配一个缓冲区哪个线程要分配内存,就现在这个区域上分配只有在这个缓冲区用完需要重新分配時,才需要同步操作
对象内存分配完之后,虚拟机需要将分配到的内存空间都初始化为0
之后虚拟机要对对象进行必要的设置例如这个對象是哪个类的实例,如何才能找到类的元数据信息、对象的哈希吗、对象的GC分代年龄等信息这些信息存放在对象头中。
上面步骤完成後在虚拟机的角度 对象已经创建完成,但在程序员的角度对象创建才刚刚开始之后按照程序员的意愿,初始化一些值之后这个对象財真正创建完成了