对象是如何创建的?
(1)检查
虚拟机遇到new指令,上常量池中找有没有要new的这个类的符号引用,并且检查,类符号引用的类是否是否已经完成了加载,解析和初始化工作,如果没有,先执行类加载的相应过程
(2)分配
为新生对象分配内存,把一块确定大小的内存从java堆中划分出来。。
分配方式:
指针碰撞:加入内存是绝对规整的,用过的放在一边,没用过的放在一边,中间用一个边界指针加以界定,那么分配内存的方式就是让边界指针向空闲区域那边移动指定大小即可。
空闲列表:如果java堆中的内存并不是完整的,已用的和空闲的相互交错,那就没法用指针碰撞了,因此,虚拟机就必须创建一个列表,来记录哪些内存是可用的,在给对象分配内存时候,通过查找该列表,选择一个足够大的内存块分配给对象实例,并同时更新列表。这就是空闲列表的方法。
综上,选择哪种方式是根据内存的规整情况来决定,但是内存的规整情况又和所采用的垃圾回收器是否带有压缩整理功能决定的
(3)并发处理
在分配的时候,由于创建对象的频率比较高,然而操作的都是同一个堆,肯定是线程不安全的,因此,需要同步处理
同步处理的方式有两种:
a.采用CAS配上失败重试保证更新操作原子性(个人理解就是,分配完了找一下,看看有哪些更新操作失败了,再重新分配一次)
b.把内存分配的动作按照线程划分在不同空间上进行,即就是每个线程在java堆中预先分配一小块的内存,称为本地线程分配缓冲(TLAB),哪个线程要分配内存,就在哪个线程的TLAB上分配,只有在TLAB用完并重新分配的时候,才需要重锁定java堆
(4)内存空间初始化为零值(保证对象的实例字段在java中不用赋值就能用)和给对象设置对象头,包括该对象是哪个类的实例,如何才能找到类的元数据信息,对象的哈希码,对象的GC分代年龄信息等
(5)初始化数据
虚拟机执行<init>方法,把对象按照程序员的意愿进行初始化
这样,,一个完整可用的对象才算完全生产出来