虚拟机遇到一条new指令时:
1.检查这个指令的参数是否能在常量池中定位到一个类的符号引用;
检查这个符号引用的类是否已被加载、解析和初始化过。
(如果没有,那必须先执行相应的类加载过程)
2.为新生对象分配内存。
对象所需的内存大小在类加载完成后便可以确定,把一块确定大小的内存从Java堆中划分出来。
如果Java堆中内存是绝对规整的,所有用过的内存都放在一边,空闲的内存放在另一边,中间放着一个指针作为分界点的指示器,那所分配的内存仅仅就是把指针指向空闲空间那边挪动一段与对象大小相等的距离,这种分配方式称为“指针碰撞”。
如果Java堆中内存并不是规整的,虚拟机就必须维护一个列表,记录哪一块内存是可用的,这种分配方式称为“空闲列表”。
Serial、ParNew等带Compact过程的收集器时,一般系统采用的是指针碰撞。
CMS这种基于Mark-Sweep算法的收集器,一般采用空闲列表。
考虑对象创建频繁,并发修改指针指向并不安全,有两种解决方案:第一种是实际上虚拟机采用CAS配上失败重试的方式保证更新操作的原子性;第二种是把内存分配的动作按照线程划分在不同的空间之中进行,即每个线程在Java堆上预先分配一小块内存。
3.内存分配完成以后。虚拟机将分配到的内存空间都初始化为零值,不包含对象头。
这一操作保证了对象的实例字段在Java代码中不赋初始值就可以直接使用。
4.对对象进行必要的设置。
这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希码、对象的GC分代年龄等等,这些信息存放在对象的对象头之中。
5.执行new指令之后会接着执行<init>方法。