面试题:JVM在Java堆中对对象的创建、内存结构、访问方式
可柔可刚,点赞则柔,白嫖则刚!死鬼~~~看完记得给我来个三连哦!
一、对象创建过程
1、检查类是否已被加载
JVM遇到new指令时,首先会去检查这个指令参数能否在常量池中定位到这个类的符号引用,检查这个符号引用代表的类是否已被加载、解析、初始化,若没有,则进行类加载
2、为新对象分配内存
类加载检查后,JVM为新对象在堆内存中分配空间,内存大小在类加载完成后便可确定。内存分配方式有以下几种:
1)指针碰撞(Bump the Pointer):若堆内存规整的,已用的和空闲的各占一边,分配内存就是把指针作为分界点,指针往空闲的一边移动对象大小的空间。
2)空闲列表(Free List):若堆内存不规整,JVM必须维护一个记录可用内存块的列表,分配内存时,把列表中一块空间分配给对象,并更新表记录。
以上两种在并发情况下,存在线程安全问题,在给对象A分配内存时,指针还没来得及修改,对象B又同时使用原来的指针来分配内存。解决方案有两种:
1)给分配内存的动作同步处理:JVM使用CAS+失败重试,保证更新操作的原子性。
2)本地线程分配缓冲(TLAB Thread Local Allocation Buffer):给每个线程在堆内存中预先分配已小块内存,在需要分配内存的线程的TLAB上分配,TLAB用完并分配新的TLAB时,才同步锁定。JVM通过设置 -XX:+UseTLAB来开启。
3、将分配到的内存都初始化为零值(不含对象头)
保证了对象的实例字段在java代码中不赋初始值就可以直接使用。如果使用TLAB,这一步可提前到TLAB分配时进行。
4、对对象进行其他必要的设置
如设置对象头的内容
5、执行java代码中<init>方法进行初始化
以上4步完成后,对于JVM来说,新的对象已经产生了,但是对于java程序来说,对象才刚刚开始创建。
二、对象的内存结构
1、对象头
1.1 标识字段 Mark Work
用于存储对象自身的运行时数据,如HashCode,GC分代年龄,锁状态标志等
1.2 类型指针 Klass Pointer
对象指向它的类型元数据的指针,JVM通过这个指针确定该对象属于哪个类的实例
如果对象是一个数组,对象头中还要有一块用于记录数组长度的数据,因为数组长度是不确定的,无法通过元数据中的信息推断数组大小。
2、实例数据
对象实际存储的有效信息,即代码中定义的字段和父类继承下来的,存储顺序受到JVM分配策略参数(-XX:FieldAllocationStyle)和代码中字段定义顺序影响
3、对齐填充
不是必然存在,仅仅是起占位符作用;由于HotSpot虚拟机的自动内存管理系统要求任何对象大小都必须是8字节的整数倍,对象头被设计成正好是8字节的整数倍,因此实例数据部分没有对齐8字节的整数倍的话,就通过对齐填充来补全。
三、对象的访问方式
java程序是通过java栈中的reference数据来操作堆中的具体对象
1、句柄访问
java堆中划分一块内存作为句柄池,栈上的reference存的是对象的句柄地址,句柄池中包含对象实例数据和类型数据的地址信息。
优点:垃圾收集移动对象时,只改变句柄中实例数据指针,而reference本身不需要修改。
2、直接访问
直接指针访问,reference存的直接是对象的地址。不需要多一次间接访问的开销。
优点:速度快,节省一次指针定位的时间开销。
HotSpot虚拟机主要使用直接访问进行对象访问。

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
利用bootstrap模态框+ajax对数据进行增删改
接上篇:1.数据库改动,添加外键,多对多model.py from django.db import models # Create your models here. GENDER = { ('male', '男'), ('female', '女'), } class User(models.Model): username=models.CharField(max_length=32,null=False) password=models.CharField(max_length=32,null=False) email = models.EmailField(null=True) name = models.CharField(max_length=20, null=True) tel = models.CharField(max_length=20, null=True) gender = models.CharField(max_length=20, choices=GENDER) birthday = models.DateTimeField(null=True) clas...
- 下一篇
关于ssd存储原理的介绍
众所周知SSD的读写速度远比hdd磁盘要快,理解ssd的工作原理使我们开发处高效储存方案。 linux 相关指令 fstrim--fstab--verbose##回收(discard)文件系统上对应磁盘未使用的块 blkdiscard/dev/nvme1n1##回收并擦除(discard)整个SSD块设备 wipefs-a/dev/nvme1n1##擦除(erase)文件系统的签名 fstrim fstrim命令实际可以视为手动对SSD磁盘下发TRIM指令。使用-v参数可以直接查看TRIM回收擦除空间的大小。fstrim是针对已挂载的文件系统的SSD分区 root@xxxx:~#fstrim--help Usage: fstrim[options]<mountpoint>Discardunusedblocksonamountedfilesystem. Options: -a,--alltrimallsupportedmountedfilesystems -A,--fstabtrimallsupportedmountedfilesystemsfrom/etc/fstab -...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- CentOS8安装MyCat,轻松搞定数据库的读写分离、垂直分库、水平分库
- CentOS6,7,8上安装Nginx,支持https2.0的开启
- CentOS关闭SELinux安全模块
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- 2048小游戏-低调大师作品
- SpringBoot2整合Redis,开启缓存,提高访问速度
- SpringBoot2更换Tomcat为Jetty,小型站点的福音
- Eclipse初始化配置,告别卡顿、闪退、编译时间过长
- Docker安装Oracle12C,快速搭建Oracle学习环境
- SpringBoot2编写第一个Controller,响应你的http请求并返回结果