一.对象创建底层步骤细节
1.虚拟机遇到一条new指令时,首先检查这个对应的类能否在常量池中定位到一个类的符号引用。
2.判断这个类是否已被加载、解析和初始化。
3.为这个新生对象在java堆中分配内存空间,其中java堆分配内存空间的方式主要有以下两种
(1)指针碰撞
a.分配内存空间包括开辟一块内存和移动指针两个步骤
b.非原子步骤可能出现并发问题,java虚拟机采用CAS配上失败重试的方式保证更新操作的
(2)空闲列表
a.分配内存空间包括开辟一块内存和移动指针两个步骤
b.非原子步骤可能出现并发问题,java虚拟机采用CAS配上失败重试的方式保证更新操作的
4.将分配到的内存空间都初始化为零值
5.设置对象头相关数据
a.GC分代年龄
b.对象的hash码hashCode
c.元数据信息
6.执行对象<init>方法
二.Java对象内存布局
1.对象头(Header)
用于存储对象的元数据信息,包括对象运行时数据和类型指针。
Mark Word部分存储对象自身的运行时数据如哈希值、GC分代年龄、锁状态标识、线程持有的锁、偏向锁ID、偏向锁时间戳。
类型指针指向它的类元数据的指针,用于判断对象属于哪个类的实例
2.实例数据(Instance Data)
实例数据存储的是真正有效数据
如各种字段内容,父类定义的变量会出现在子类定义的变量的前面
3.对齐填充(Padding)
对齐填充部分仅仅起到占位符的作用,并非必须
三.对象访问定位方式
1.当我们在堆上创建一个对象实例后,就要通过虚拟机栈中的reference类型数据来操作堆上的对象。现在主流的访问方式有两种(HotSpot采用的是第二种)
(1).使用句柄访问对象,即reference中存储的是对象句柄的地址。而句柄中包含了对象实例数据与类型数据的具体地址信息,相当于二级指针
(2).直接指针访问对象,即reference中存储的就是对象地址,相当于一级指针。
2.上面两种对比
(1)垃圾回收分析:方式一当垃圾回收移动对象时,reference中存储的地址是稳定的地址,不需要修改,仅需要修改对象句柄的地址;方式二垃圾回收时需要修改reference中存储的地址。
(2)访问效率分析:方式二优于方式一,因为方式二只进行了一次指针定位,节省了时间开销,而这也是Hotspot采用的实现方式。