深入理解Java虚拟机(二)——对象的创建

723 阅读3分钟

对象的创建过程

当我们使用new关键字去创建类对象的时候,虚拟机会发生一系列的操作:


1.检查准备创建的对象所属类是否已被加载,若没有,则先加载该类;若加载完毕,进行下一步。


2.为新生对象分配空间。类加载完成后,对象所需的空间是完全确定下来的,分配空间就是在内存中划出一小块空间分配给对象。分配方式有两种:

  • ●指针碰撞:当我们使用的垃圾收集算法会导致已被使用在内存在一边,空闲的内存在另一边时(复制算法、标记-整理算法),可通过一个指针往空闲内存方向移动一定的距离,即可实现为新生对象分配内存。
  • ●空闲列表:当我们使用的垃圾收集算法会导致已使用的内存和空闲的内存相互交错时(标志-清除算法 ),虚拟机会把空闲内存的地址和大小记录在空闲列表中,在需要分配对象内存时可通过空闲列表查找到空闲内存进行分配。

3.将对象新分配到的空间初始化为零值,保证对象不赋初始值就可以直接使用。


4.设置对象头的信息。


5.从虚拟机的角度讲,新对象已经生成;从Java程序的角度讲,对象创建才刚开始——对象的构造函数还未被调用。因此,接下来要做得是调用对象的构造函数进行初始化。

对象的内存布局

在hotspot虚拟机中,对象在内存里可分为3个区域:对象头、实例数据、对齐填充。

1.对象头

对象头存储了两部分信息:

  • ●存储对象自身的运行数据,如哈希吗、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳等。这部分数据被官方称之为“Mark Down”,又因为这部分存储空间有限,无法把所有信息都囊括其中,于是被设置为一个非固定结构以便存储尽量多的信息,它会根据对象的状态复用自身存储空间。
  • ●存储对象的类型指针,该指针指向对象的类元数据,通过这个指针可确定对象是哪个类的实例。
  • ●如果对象是数组,对象头中还存储了数组的长度。

2.实例数据

这部分存储了对象的成员变量,包括了父类和子类的成员变量。

3.对齐填充

这部分空间本身并没有多大作用,只是对象的存储空间被要求为8字节的整数倍,当对象大小不符合要求时,对齐填充可解决这个问题。

对象的访问方式

对象的访问方式有两种:句柄访问、直接访问。

  • ●句柄访问:Java堆中会划分出一块内存用于充当句柄池,对象引用存储了指向句柄池的地址,而句柄池里则存储了对象实例数据地址和对象类型数据地址
  • ●直接访问:对象引用存储的就是对象在Java堆中的地址。

  • 比较:句柄访问的好处在于当对象发生移动时,句柄池中的对象实例指针发生更改,而对象引用不需要改变;直接访问的好处在于速度更快。hotspot虚拟机选择了直接访问的方式。