深入理解JVM(一)——对象的创建

13 篇文章 0 订阅

对象的创建

对象的创建,在语言层面上,通常只是new这个关键字而已。(本章所讨论的对象限于普通Java对象,不包括数组和 Class对象)虚拟机遇到new指令时:

  1. 检查这个指令的参数是否能在常量池定位到一个类的符号引用。
  2. 检查这个符号引用代表的类是否已被加载,解析,初始化过。
  3. 如果没有必须执行相应的类加载过程

类加载检查后,虚拟机会为新生对象分配内存。对象所需的内存在类加载时可完全确定,为对象分配空间等同于把一块确定大小的内存从Java堆中拿出来。

规整的内存空间可用指针碰撞法分配
非规整的内存空间用空闲列表法分配(虚拟机维护一张列表记录哪些内存块是可用的)

内存分配在并发情况下的线程安全

  1. 对分配内存空间的动作进行同步处理——实际上虚拟机采用的CAS配上失败重试的方式保证更新操作的原子性
  2. 把内存分配的动作按照线程划分在不同的空间上进行,即每个线程在Java堆上预先分配一小块内存(Thread Local Allocation Buffer,TLAB),虚拟机是否启动TLAB由参数-XX:+/-UserTLAB设定

内存分配完成,虚拟机将分配到的内存空间初始化为零值(不包括对象头)

虚拟机对对象进行必要的设置

  • 这个对象是哪个类的实例
  • 如何才能找到类的元数据信息
  • 对象的哈希码
  • 对象的GC分代年龄等信息

这些信息存放在对象的对象头中

执行init方法
对象空间分配完成,但是所有的字段都是零。一般来说由字节码中是否跟随invokespecial指令决定,执行玩new指定后,是否执行init方法

对象的内存布局

包括3个区域,对象头,实例数据,对齐填充

  • 对象头

    存储自身运行时的数据,如哈希码,GC分代年龄,锁状态标志,线程持有的锁,偏向线程ID,偏向时间戳等,长度为32bit/64bit,根据虚拟机的位数决定。官方成为Mark Word,被设计为一个非固定的数据结构,以便在极小的空间存储更多的信息。
    存储类型指针,即指向他的类元数据(确定该对象是哪个类的实例),数组还需要记录长度

  • 数据实例

    对象真正存储的有效信息,程序代码定义的各种字段内容,包括父类继承的。这部分的存储顺序会受到虚拟机的分配策略参数和字段定义顺序影响。

  • 对齐填充

    不是必然存在,仅仅是占位符的作用。HotSpot VM内存管理系统要求对象的起始地址必须是8字节的整数倍,所以对象也必须是8字节的整数倍。

对象的访问定位

建立对象是为了使用对象,Java程序需要通过栈上的reference数据来操作堆上的具体对象。
Java虚拟机并未规定具体的定位方式,主流的访问方式有两种

  • 使用句柄

Java划分一块内存作为句柄池,reference中存的是对象的句柄地址,句柄中包含对象的实例数据与类型数据各自具体的地址。(GC时对象被移动,reference不需要修改)

  • 直接指针

Java堆对象的布局必须考虑如何放置访问类型数据的相关信息,而reference中存储的就是对象的地址。(速度快,节省一次指针定位的开销)(HotSpot VM)

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值