深入理解Jvm(二)--对象

本文详细介绍了Java对象的创建过程,包括JVM如何在堆内存中分配内存,以及在并发情况下保证线程安全的两种策略:CAS+失败重试和线程本地分配缓冲区(TLAB)。接着,讨论了对象的初始化步骤,如设置对象头信息并执行初始化方法。此外,还阐述了对象内存布局的三个部分——对象头、实例数据和对齐填充,以及它们各自的作用和实现细节。
摘要由CSDN通过智能技术生成

1、对象的创建:(普通对象,不包过数组和Class对象)

  • new字节码指令:当Jvm遇到字节码new指令时,首先检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载、解析和初始化过。如果没有必须执行类的加载过程。(双亲委派机制)
  • 计算创建对象所需要的内存:类加载检查通过后,Jvm将为新生的对象分配内存,对象所需内存的大小在类加载完成后便完全确定。在堆内存中就是将一块确定大小的空闲的内存分配出来。

2、对象的内存分配:

  • 如果内存时整齐连续的:如果垃圾回收机制是带有空间压缩整理的方式,那么堆内存空间就是整齐的,在给对象分配内存时就是将内存指针向空闲内存的方向偏移一下,这种分配内存的方式称之为 “指针碰撞分配对象内存”。
  • 如果堆内存不是整齐的:Jvm就需要维护一张内存空闲列表,在分配内存时找一块能够满足当前对象需要的内存分配给该对象,同时记录在空间列表中。这种分配内存的方式称之为 “空闲列表分配对象内存”

3、分配内存并发情况下保证线程安全问题:

  • 问题:无论是采用指针碰撞还是空闲列表的方式分配内存,都可能存在线程安全问题,因为分配的动作和内存记录更新的动作并不是原子性的操作
  • 解决:方案一分配内存空间和更新内存记录的动作进行同步处理。Jvm是通过CAS+失败重试的方式保证分配和更新操作的原子性。
  • 解决:方案二将内存分配的动作按照线程划分不同的空间之中进行。即在堆内存中为每一个线程分配一块小空间,称之为线程分配缓冲TLAB,Jvm是否使用TLAB可以通过参数配置。

 4、对象的初始化:

  • 在对象头中记录一些对象的基本信息。
  • 调用Class文件中的init方法,对对象进行初始化。

5、对象的内存布局:对象头 Mark Word

  • 对象头是一个动态定义的数据结构,大小可变,32位或者64位。Hotspot对象头中包括两类信息:
  • a.存储对象自身运行时数据,如:哈希码、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳等。锁的标识位是两个bite,0和1的组合共五种。
  • b.类型指针:对象指向它的类型元数据的指针,通过这个指针来确定该对象是那个类的实例。如果对象是数组,需要在对象头中记录该对象的数组长度。Jvm可以通过普通Java对象的元数据信息确定Java对象的大小。

6、对象的内存布局:实例数据

  • 存储对象实例中的所有数据:程序中定义的各种类型的字段内容,无论是父类继承还是自身数据都需要记录下来。
  • 字段的分配策略:相同宽度的数据类型会被分配存储在一起。

7、对象的内存布局:对齐填充

  • 不是必然存在:对齐填充的作用是占位符的作用。HotSpot虚拟机的自动内存管理系统要求对象的大小必须是8的倍数,对象头的设计已经是8的倍数,所以如果实例数据不满足8的倍数,就需要对齐填充来完成补全

 

 

 

 

 

 

 

 

 

 

 

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值