jvm中对象创建过程分为五部分
1:类加载检测
2:对象分配
3:初始化零值
4:设置对象头
5:执行int方法
类加载检查
当虚拟机遇到一个new 指定后 执行的步骤:会通过类加载器进行加载,
1:首先会在方法区(元空间)中的常量池,定位这个类的符号引用在不在。
2:如果在,检测这个符号引用,所代表的类,是否已经加载过,解析过,初始化过
3:如果不在(相当于没有加载过),那就执行类加载过程
4:确定对象所需要的分配内存的大小
5:类加载检测通过后,为新生对象分配内存
分配内存
给新生对象,在堆内存中分配一块区域,不同的jvm虚拟机,堆得内存规整情况是不同的,所以分配方式有一定区别(堆内存是否规整,是有jvm所采取的垃圾回收器是都带有压缩整理功能决定的)
1:规整内存(指针碰撞方式),用过的内存放在一块,没用的放在一块,中间用指针做标识,分配的时候指针向空闲内存移动。把新生对象在空闲内存分配一块区域
2:不规整内存(空闲列表):用过的内存和没用过的内存,是相互交错的。jvm会维护一个表,记录那些内存是可用的。 然后给新生对象分配一个足够大的内存。并更新表内记录。
并发时内存分配:
因为对象创建是很频繁的,并发时,就会存在线程安全。
列:程序中,如果同时创建A和B,底层给A分配内存,指针还没有修改,同时对B使用原来指针分配内存。就会出现问题
目前我知道的是两种方式来解决:
一种是CAS(同步处理):CAS指令去更新变量结合失败重试,失败的线程将会不断重新尝试,直到给对象分配成功
一种是 本地线程分配缓冲(TLAB):既 每个线程在堆中预先分配一小块内存
初始化:内存分配完成,虚拟机把分配内存空间的数据类型,都初始化为零值,int类型0 String类型null
设置对象头:
对象头包括:MarkWord和ClassPointer
MarkWord:包括对象gc分代年龄,gc标记(三色标记),锁等
ClassPointer(类型指针):指向对象对应的内存地址
对象填充:将对象内存填充为8的倍数,目的是alignment,它允许以一些空间为代价加快内存访问。如果数据未对齐,则处理器需要在加载内存后进行一些转换才能访问它
类指针压缩:压缩对象头中的ClassPoint,由把字节压缩4个字节,提高内存利用率
执行init方法:
对象初始化,属性赋值,执行构造方法