java JLabel改变大小后如何刷新_jvm大局观之内存管理篇(二):当java中new一个对象,背后发生了什么...

前言

本篇是java内存区域管理系列教程之一 java创建对象的过程, 教程全系列内容如下

1.java内存区域的划分

2.如何向java内存区域存值

3.如何判断java内存区域中的那些值是无用的

4.如何将java内存区域中无用的值清理掉

5.java内存区域的垃圾清理器

6.低延时垃圾收集器-Shenandoah

今天我们谈谈 当java中new一个对象,背后发生了什么

概括说来,就是 先后执行类加载,分配内存,初始化零值,设置对象头,初始化对象,对象引用入栈

看完本篇文章,读者将能够回答以下问题

1.当java中new一个对象,背后发生了什么

2.java内存分配中的指针碰撞和空闲列表分配是什么意思

3.多线程下对象的创建会存在什么问题

码字不易,赠人玫瑰,手留余香,点赞的读者据说都找到了失散已久的妹妹

224af73a825808febd5f96efc25b2004.png
等待领养的妹妹 :)

正文

我们以最常用的虚拟机HotSpot和最常用的内存区域Java堆为例,深入探讨一下HotSpot虚拟机在Java堆中对象分配全过程

29445d541393c2836c28b4d189dafc48.png

一. 新生对象所属类的加载检查

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

二. 为新生对象分配内存

在类加载检查通过后,接下来虚拟机将为新生对象分配内存。

对象所需内存的大小在类加载完成后便可完全确定,为对象分配空间的任务实际上便等同于把一块确定大小的内存块从Java堆中划分出来

2.1 如何分配到一块空闲的内存

2.1.1 指针碰撞分配 。假设Java堆中内存是绝对规整的,所有被使用过的内存都被放在一 边,空闲的内存被放在另一边,中间放着一个指针作为分界点的指示器,那所分配内存就仅仅是把那 个指针向空闲空间方向挪动一段与对象大小相等的距离,这种分配方式称为“指针碰撞”(Bump The Pointer)。

c281874721341f012e3c7efb3eb3191c.png

2.1.2 空闲列表分配

但如果Java堆中的内存并不是规整的,已被使用的内存和空闲的内存相互交错在一起,那 就没有办法简单地进行指针碰撞了,虚拟机就必须维护一个列表,记录上哪些内存块是可用的,在分 配的时候从列表中找到一块足够大的空间划分给对象实例,并更新列表上的记录,这种分配方式称 为“空闲列表”(Free List)

2.1.3 垃圾收集器是否带有空间压缩整理决定是上面两种分配方式中的一种

选择哪种分配方式由Java堆是否规整决定,而Java堆是否规整又由所采用 的垃圾收集器是否带有空间压缩整理(Compact)的能力决定。 因此,当使用Serial、ParNew等带压缩整理过程的收集器时,系统采用的分配算法是指针碰撞,既简单又高效; 而当使用CMS这种基于清除(Sweep)算法的收集器时,理论上就只能采用较为复杂的空闲列表来分配内存。

三. 新生对象各属性初始化零值

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

四. 设置新生对象的对象头

接下来,Java虚拟机还要对对象进行必要的设置,例如这个对象是哪个类的实例、如何才能找到 类的元数据信息、对象的哈希码(实际上对象的哈希码会延后到真正调用Object::hashCode()方法时才 计算)、对象的GC分代年龄等信息。这些信息存放在对象的对象头(Object Header)之中。

五. 为新生对象执行init方法

在上面工作都完成之后,从虚拟机的视角来看,一个新的对象已经产生了。

但是从Java程序的视 角看来,对象创建才刚刚开始——构造函数,即Class文件中的()方法还没有执行,所有的字段都 为默认的零值,对象需要的其他资源和状态信息也还没有按照预定的意图构造好。

一般来说new指令之后会接着执行 ()方法,按照程序员的意愿对对象进行初始化,这样一个真正可用的对象才算完全被构造出来。

六. 将新生对象的引用入栈

如果有类似于Student s =new Student();形式的引用的话,在栈区定义Student类型引用变量s,然后将堆区对象的地址赋值给它.


扩展

扩展1:new对象过程案例:

Student类代码如下

public 

Student s = new Student()做了哪些事情?

1、把Student. class文件加载到内存

2、在栈内存给s变量开辟一个空间

3、在堆内存为学生对象申请一个空间

4、给成员变量进行默认初始化。null,0

5、给成员变量进行显示初始化。林青霞,27

6、通过构造方法给成员变量进行初始化。刘意,30

7、数据初始化完毕,然后把堆内存的地址值赋值给栈内存的s变量。

24d490776b51d5b0720dfde7d6cd819e.png

扩展2: 创建对象指令重排序问题:

new一个对象的简单分解动作

  1. 分配对象的内存空间
  2. 初始化对象
  3. 设置引用指向分配的内存地址

其中2和3两步间会发生指令重排序,导致多线程时,如果在初始化之前访问对象,则会出现问题,单例模式的双重检测锁模式正是会存在这个问题。可以使用volatile来禁止指令重排序解决问题

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值