JVM创建对象过程

<div class="iteye-blog-content-contain" style="font-size: 14px"></div>

 1、找到类对象

<!--?xml version="1.0" encoding="UTF-8" standalone="no"?-->
     当碰到new关键字的时候,虚拟机要去方法区的常量池中寻找,是否能找到这个类的符号引用,如果没有找到,那么必须先执行类的加载过程、解析和初始化。对于继承而言,同样需要同时找到其父类(从下到上的顺序去查找)
 
2、分配内存并初始化零值
     类加载完成后就需要在Java堆中分配指定大小的内存,这个大小是在类加载完成后确定的。
     假设Java堆中的内存是绝对规整的,使用中的内存放一边,没使用的放在另一边,中间放着一个指针区分,那么在对象分配内存的时候,指针向空闲的那边挪动一段与类实例对象大小相同的区域,这种叫做指针碰撞。但是并非所有的GC算法在内存回收后都会把内存区域整理一遍,这个时候内存是零散的,这个时候就需要虚拟机去维护一个可用内存列表了,在分配内存的时候找出一块足够大的空间来分配。
     这里还需要注意的是,对象在分配内存的时候的 原子性的问题:假设我一个对象正在分配内存,指针还没来得及确定距离,下一个对象就来分配,并使用指针原来的位置,这个时候就会出问题。
     解决这个问题的方案有两种:1.对对象内存的分配使用同步的方式处理,就是一个对象正在分配内存的时候,内存指针不允许其他对象分配使用,2.另外一种方法就是利用缓存,在每一个线程设置本地线程分配缓冲TLAB,在分配对象内存的时候,先在TLAB中分配,如果TLAB中的内存不够了,那么就再同步锁定,重新分配。
     内存分配完成后,类对象里面的属性都初始化为0,或者是Null。以保证对象实例字段在不赋值的情况下也能使用。
 
3、设置对象的对象头
     对象头中会存放,这个对象是哪个类的实例,如何才能得到类的元数据信息、对象的哈希值,GC年龄带信息等等。
 
4、对象初始化
     这个时候开始执行init方法,按照程序员的意愿把对象里面的属性进行合理的初始化,然后一个对象才完整的诞生。对于继承的对象,会从父类开始向下执行初始化,这个初始化,并不是对父类也创建对象,然后初始化哦。
 
 
 
 
 
 
 
 
 
 

猜你喜欢

转载自we-are-here.iteye.com/blog/2262399