Java的对象是在JVM中是如何创建的?

一、对象的实例化

创建对象的方式:

  • 通过new关键字(或者一些设计模式下的获取对象方法)
  • 通过Class的newInstance(),也就是反射的方式,但是要满足权限是public和无参构造器
  • 通过Constructor的newInstance(args[]):也是反射的方式,但是可以调用空参或者带参的构造器,权限没有要求
  • 使用clone():需要该类实现Cloneable接口中的clone()方法
  • 使用反序列化:例如从文件中获取对象流转成对象
  • 使用第三方库Objenesis等

创建对象的步骤:

  1. 判断对象对应的类的加载:
    • 若对象对应的类已经经过加载、链接、初始化则进入第下一步,若没有则进行该系列操作再进入下一步。
  2. 为对象分配内存:
    • 计算对象占用的空间大小,在堆中划分内存:
      • 若堆内存规整(使用标记整理算法进行GC的),采用指针碰撞法(也就是用指针start-top-end三个指针来区别已分配和未分配的空间)来分配空间。
      • 若堆内存不规整(使用标记清除算法进行GC会残留下空间碎片),在虚拟机中创建一个空闲列表,该列表记录堆空间中哪些内存块是已分配,哪些内存块是未分配的,然后在未分配的内存块中找到一块足够创建该对象的内存块来分配。
    • 处理分配安全问题的方式:
      • 可以采用CAS、区域加锁来保证原子性或者是使用TLAB块来给分配。
  3. 对对象进行默认初始化:
    • 例如int类型的对象赋为0,boolean对象赋为false,类对象赋为null等等。
  4. 设置对象头(后续介绍):
    • 设置对象的所属类、HashCode、GC信息、以及锁信息等数据存储在对象头中
  5. 执行方法进行初始化:
    • 根据类构造器或者是代码块进行初始化。

对象的内存布局:

  • 对象头:
    • 运行时元数据:哈希值、GC分代年龄、锁状态标志、线程持有锁、偏向线程ID、偏向时间戳
    • 类型指针:指向类元数据InsatanceKlass,确定该对象所属的类型
  • 实例数据:
    • 对象中真正存放的数据,包括对象的各种类型的字段(包括从父类继承下来的各种字段)
    • 规则:先加载父类的变量、相同长度的字段分配在一起、如果设置CompactFields参数,可以将子类的窄变量插入到父类留下的缝隙中。
  • 对齐填充:
    • 不一定存在,起占位符的作用。

图示:在这里插入图片描述

对象的访问定位:

句柄访问:
  • 图示:在这里插入图片描述

  • 优点:当对象的实例位置发生改变的时候只需要改变句柄池中的指针引用。

  • 缺点:耗费空间创建句柄池,访问效率相对慢。

直接指针(Hotspot采用):
  • 图示:在这里插入图片描述

  • 优点:访问速度快,相对句柄占用内存小。

  • 缺点:当对象位置发生变化的时候需要重新将引用指向对象的实例。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值