jvm中对象创建、内存布局以及访问定位

2023-09-18 23:30:22

对象创建

Java语言层面,创建对象通常(例外:复制、反序列化)仅仅是一个new关键字即可,而在虚拟机中,对象(限于普通Java对象,不包括数组和Class对象等)的创建又是怎样一个过程呢?

① Java虚拟机遇到一条字节码new指令时,首先将去检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载、解析和初始化过。如果没有,那必须先执行相应的类加载过程

类加载检查通过后,接下来虚拟机将为新生对象分配内存。对象所需内存的大小在类加载完成后便可完全确定,为对象分配空间实际上便等同于把一块确定大小的内存块从Java堆中划分出来。此时有两种实现情况:

  1. 假如Java堆中内存是绝对规整的被使用过的内存都被放在一边,空闲的内存被放在另一边中间放着一个指针作为分界点的指示器,那所分配内存就仅仅是把那个指针向空闲空间方向挪动一段与对象大小相等的距离,这种分配方式称为“指针碰撞”(Bump ThePointer)
  2. 假如Java堆中的内存并不是规整的已被使用的内存和空闲的内存相互交错在一起,虚拟机就必须维护一个列表,记录上哪些内存块是可用的,在分配的时候从列表中找到一块足够大的空间划分给对象实例,并更新列表上的记录,这种分配方式称为“空闲列表”(Free List)

选择哪种分配方式由Java堆是否规整决定,而Java堆是否规整又由所采用的垃圾收集器是否带有空间压缩整理的能力决定。【因此选择哪种分配方式,主要由垃圾收集器决定】

使用Serial、ParNew等带压缩整理过程的收集器时,系统采用的分配算法是指针碰撞,既简单又高效;而当使用CMS基于清除(Sweep)算法的收集器时,理论上就只能采用较为复杂的空闲列表来分配内存。

对象创建在虚拟机中是非常频繁的行为,即使仅仅修改一个指针所指向的位置,在并发情况下也并不是线程安全的,解决这个问题有两种可选方案:

  1. 对分配内存空间的动作进行同步处理——实际上虚拟机是采用CAS配上失败重试的方式保证更新操作的原子性
  2. 把内存分配的动作按照线程划分在不同的空间之中进行,即每个线程在Java堆中预先分配一小块内存,称为本地线程分配缓冲(Thread Local AllocationBuffer,TLAB),哪个线程要分配内存,就在哪个线程的本地缓冲区中分配,只有本地缓冲区用完了,分配新的缓存区时才需要同步锁定虚拟机是否使用TLAB,可以通过-XX:+/-UseTLAB参数来设定

内存分配完成之后,虚拟机必须将分配到的内存空间(但不包括对象头)都初始化为零值,如果使用了TLAB的话,这一项工作也可以提前至TLAB分配时顺便进行。这步操作保证了对象的实例字段在Java代码中可以不赋初始值就直接使用,使程序能访问到这些字段的数据类型所对应的零值。

Java虚拟机还要对对象进行必要的设置,例如这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希码(实际上对象的哈希码会延后到真正调用Object::hashCode()方法时才计算)、对象的GC分代年龄等信息。这些信息存放在对象的对象头(Object Header) 之中。根据虚拟机当前运行状态的不同,如是否启用偏向锁等,对象头会有不同的设置方式。

此时,从虚拟机的视角来看,一个新的对象已经产生了。从Java程序的视角看来,new指令之后会接着执行()方法,按照程序员的意愿对对象进行初始化,这样一个真正可用的对象才算完全被构造出来。

jvm对象创建过程

对象内存布局

对象在堆内存中的存储布局可以划分为三个部分:对象头(Header)实例数据(Instance Data)对齐填充(Padding)

对象的对象头部分包括两类信息

  1. 第一类是用于存储对象自身的运行时数据,如哈希码(HashCode)、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳等,这部分数据的长度在32位和64位的虚拟机(未开启压缩指针)中分别为32个比特和64个比特,官方称它为“Mark Word”。对象需要存储的运行时数据很多,实际上已经超出了32、64位Bit map结构所能记录的最大限度,Mark Word被设计成一个有着动态定义的数据结构,以便在极小的空间内存储尽量多的数据,根据对象的状态复用自己的存储空间
  2. 另外一部分是类型指针,即对象指向它的类型元数据的指针,Java虚拟机通过这个指针来确定该对象是哪个类的实例。

实例数据部分是对象真正存储的有效信息,即在程序代码里面所定义的各种类型的字段内容,无论是从父类继承下来的,还是在子类中定义的字段都必须记录起来。这部分的存储顺序会受到虚拟机分配策略参数(-XX:FieldsAllocationStyle参数)和字段在Java源码中定义顺序的影响。

如果HotSpot虚拟机的+XX:CompactFields参数值为true(默认就为true),那子类之中较窄的变量也允许插入父类变量的空隙之中,以节省出一点点空间

对齐填充,这并不是必然存在的,也没有特别的含义,它仅仅起着占位符的作用

HotSpot虚拟机的自动内存管理系统要求对象起始地址必须是8字节的整数倍,即任何对象的大小都必须是8字节的整数倍。对象头部分已经被精心设计成正好是8字节的倍数(1倍或者2倍),如果对象实例数据部分没有对齐的话,就需要通过对齐填充来补全。

对象的访问定位

创建对象是为了后续使用该对象,Java程序会通过栈上的reference数据来操作堆上的具体对象。

对象访问方式也是由虚拟机实现而定的,主流的访问方式主要有使用句柄直接指针两种:

  • 句柄:Java堆中将可能会划分出一块内存来作为句柄池,reference中存储的就是对象的句柄地址,而句柄中包含了对象实例数据与类型数据各自具体的地址信息。

使用句柄来访问的最大好处就是reference中存储的是稳定句柄地址,在对象被移动(垃圾收集时移动对象是非常普遍的行为)时只会改变句柄中的实例数据指针,而reference本身不需要被修改。

image-20230918232406004

  • 直接指针访问:reference中存储的直接就是对象地址,如果只是访问对象本身的话,就不需要多一次间接访问的开销。

image-20230918232542071

更多推荐

2023华为杯研究生数学建模C题分析

完整的分析查看文末名片获取!问题一在每个评审阶段,作品通常都是随机分发的,每份作品需要多位评委独立评审。为了增加不同评审专家所给成绩之间的可比性,不同专家评审的作品集合之间应有一些交集。但有的交集大了,则必然有交集小了,则可比性变弱。请针对3000支参赛队和125位评审专家,每份作品由5位专家评审的情况,建立数学模型确

【Linux】系统级文件操作&&文件系统的概念

我们在C语言都学过文件操作,例如fopen,fclose之类的函数接口,在C++中也有文件流的IO接口,那不仅仅是C/C++,python、java、go、hph等等这些语言也都有自己的文件操作的IO接口。那有没有一种统一的视角来看待这些文件操作呢?它们的底层原理到底是什么?下面我们就来好好谈一谈:目录一、Linux操

数据结构-----串(String)详解

目录前言1.串的定义相关类型2.串的储存结构顺序储存表示堆分配储存表示块链储存表示3.串的操作方式4.串的匹配算法(1)BF算法过程原理代码实现(C/C++)算法分析(2)KMP算法过程原理匹配过程:获取next数组:代码实现(C/C++)算法分析前言前面我们学习了顺序表和线性表,这两种数据结构的储存数据域可以是一个任

EasyCode整合mybatis-plus的配置

文章目录entitymapper.javamapper.xmlserviceserviceImplcontroller这篇文章不教你如何安装和使用EasyCode,只是贴出可以使用的配置。具体EasyCode的使用可以查看其它的文章。entity##导入宏定义$!{define.vm}##保存文件(宏定义)#save(

AI图像渲染

得益于AI技术的极大提升,我们可以耗费更少的计算资源和带宽来实现更清晰,细节更丰富和帧率更高的画面。什么是DLSSDLSS是一种利用深度学习的超级采样技术,可以在低分辨率的渲染下生成高分辨率的画面,从而提高游戏的性能和画质。DLSS的原理是通过在超级计算机上训练一个神经网络,来学习不同游戏和场景的图像特征,然后在GPU

Postgresql事务测试

参考一个事务中可以查询自己未提交的数据吗_最详细MySQL事务隔离级别及原理讲解!(二)-CSDN博客一个事务中可以查询自己未提交的数据吗_趣说数据库事务隔离级别与原理_weixin_39747293的博客-CSDN博客【MySql:当前读与快照读】_当前读和快照读_QuietThinking的博客-CSDN博客Pos

单片机第三季-第二课:STM32存储器、电源和时钟体系

目录1,存储器1.1,位带操作2,启动模式3,电源管理系统4,复位和时钟4.1,复位4.2,时钟1,存储器ICode总线:该总线将Cortex™-M3内核的指令总线与闪存指令接口相连接。指令预取在此总线上完成。DCode总线:该总线将Cortex™-M3内核的DCode总线与闪存存储器的数据接口相连接(常量加载和调试访

算法|图论 4

LeetCode827.最大人工岛题目链接:力扣(LeetCode)官网-全球极客挚爱的技术成长平台题目描述:给你一个大小为nxn二进制矩阵grid。最多只能将一格0变成1。返回执行此操作后,grid中最大的岛屿面积是多少?岛屿由一组上、下、左、右四个方向相连的1形成。解题思路(深度优先遍历):首先,通过深度优先遍历,

怒刷LeetCode的第7天(Java版)

目录第一题题目来源题目内容解决方法方法一:二分+贪心方法二:二分+DP第二题题目来源题目内容解决方法方法一:双指针方法二:暴力搜索方法三:排序第三题题目来源题目内容解决方法方法一:回溯算法方法二:队列方法三:递归方法四:迭代第一题题目来源2560.打家劫舍IV-力扣(LeetCode)题目内容解决方法方法一:二分+贪心

在IOS 的开发中iBeacon和BLE的区别

区分3个概念beacon、BLE、iBeaconBLE全称为BluetoothLowEnergy,蓝牙低功耗技术。现在的蓝牙技术所消耗的功耗很低,并不像传闻中不使用蓝牙需要关闭省电,手机上长期打开蓝牙所消耗的电量可以忽略不计。beacon是用于室内定位,基于BLE协议的在广播搜索上稍作变化变更了数据类型实现的一个应用场

智能井盖:提升城市井盖安全管理效率

窨井盖作为城市基础设施的重要组成部分,其安全管理与城市的有序运行和群众的生产生活安全息息相关,体现城市管理和社会治理水平。当前,一些城市已经将智能化的窨井盖升级改造作为新城建的重要内容,推动窨井盖等“城市部件”配套建设物联智能感知设备,促进现代信息技术与城市生命线深度融合,提升窨井盖安全管理效率和水平。智能井盖通过再井

热文推荐