本文讨论的对象仅限于Java对象,不包括数组和Class对象等。
那么创建是一个怎样的过程呢?
- 虚拟机遇到一条new 指令时,首先将去检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已经被加载、解析和初始化。如果没有那么必须先执行相应的类加载的过程。
总结:虚拟机先判断该类对象是否已经被加载、解析和初始化,这部分内容后面会具体讲。
- 类加载检查通过后,虚拟机将为新生对象分配内存(一般是在堆中)。内存的大小如何确定呢?因为对象是类的一个实例,那么在加载类时,对象的大小已经被确定了。
- 除了如何划分可用空间外,还需要考虑对象创建在虚拟机中是一个频繁的过程,在并发情况下如何考虑线程安全问题?
- 实际上虚拟机采用CAS(乐观锁)配上失败重试机的方式保证原子性。
- 同时在堆中为每一个线程分配一个小块空间(Thread Local Allocation Buffer, TLAB)。
- 因此只有当该线程的TLAB分配完时,新产生的对象才会分配到堆中,然后采用CAS+重试保证线程安全。
- 内存分配完后,虚拟机需要将分配到的内存空间都初始化为零值,例如:对象的实例变量等等。(不包括对象头)
- 接下来,虚拟机要对对象进行必要的设置,例如这个对象是哪个类对象的实例,如何找到类的元数据信息,哈希值,GC分代年龄等等。这些值存放在对象头中。
- 到这里对象创建才刚刚开始,然后调用invokespecial指令(相当于调用构造函数)把对象按照程序员的意愿进行初始化。
类创建的过程大致如下图所示。
【总结】
- 先判断类加载
- 然后在堆中(新生代区域)分配内存,利用CAS和重试机制保障线程安全。
- 分配内存后,初始化对象头信息。
- 调用invokespecial指令,将对象进行初始化。(相当于执行构造函数)