<JVM上篇:内存与垃圾回收篇>08-对象实例化及直接内存

简介: <JVM上篇:内存与垃圾回收篇>08-对象实例化及直接内存

8. 对象实例化及直接内存

8.1. 对象实例化

面试题

美团

对象在 JVM 中是怎么存储的?

对象头信息里面有哪些东西?

蚂蚁金服

Java 对象头有什么?

8.1.1. 创建对象的方式


d505ceb06b59d36bf0bb9da59ec2a181.png

代码演示

/**
 * @author shkstart  shkstart@126.com
 * @create 2020  17:16
 */
public class ObjectTest {
    public static void main(String[] args) {
        Object obj = new Object();
    }
}


结果分析

4e53ebdb6cc5a60090142b6efe824472.png

8.1.2. 创建对象的步骤

前面所述是从字节码角度看待对象的创建过程,现在从执行步骤的角度来分析:

aa617406c900d409b8a109657d760af6.png

1. 判断对象对应的类是否加载、链接、初始化

虚拟机遇到一条 new 指令,首先去检查这个指令的参数能否在 Metaspace 的常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已经被加载,解析和初始化(即判断类元信息是否存在)。


如果没有,那么在双亲委派模式下,使用当前类加载器以 ClassLoader + 包名 + 类名为 key 进行查找对应的 .class 文件;


如果没有找到文件,则抛出 ClassNotFoundException 异常

如果找到,则进行类加载,并生成对应的 Class 对象

2. 为对象分配内存


首先计算对象占用空间的大小,接着在堆中划分一块内存给新对象。如果实例成员变量是引用变量,仅分配引用变量空间即可,即 4 个字节大小 (引用变量只保存引用,所以大小能确定,而非引用变量大小都是固定的。)


如果内存规整:虚拟机将采用的是指针碰撞法(Bump The Point)来为对象分配内存。


意思是所有用过的内存在一边,空闲的内存放另外一边,中间放着一个指针作为分界点的指示器,分配内存就仅仅是把指针指向空闲那边挪动一段与对象大小相等的距离罢了。如果垃圾收集器选择的是 Serial ,ParNew 这种基于压缩算法的,虚拟机采用这种分配方式。一般使用带 Compact(整理)过程的收集器时,使用指针碰撞。



如果内存不规整:虚拟机需要维护一个空闲列表(Free List)来为对象分配内存。


已使用的内存和未使用的内存相互交错,那么虚拟机将采用的是空闲列表来为对象分配内存。意思是虚拟机维护了一个列表,记录上那些内存块是可用的,在分配的时候从列表中找到一块足够大的空间划分给对象实例,并更新列表上的内容。

选择哪种分配方式由 Java 堆是否规整所决定,而 Java 堆是否规整又由所采用的垃圾收集器是否带有压缩整理功能决定。


3. 处理并发问题


采用 CAS(自旋锁) 失败重试、区域加锁保证更新的原子性

每个线程预先分配一块 TLAB:通过设置 -XX:+UseTLAB参数来设定

4. 初始化分配到的内存(赋默认初始值)

所有属性设置默认值,保证对象实例字段在不赋值时可以直接使用


**举例:**对于基本数据类型byte,short,int,long,float,double为0,布尔类型为false,引用类型默认为NULL


5. 设置对象的对象头


将对象的所属类(即类的元数据信息)、对象的 HashCode 和对象的 GC 信息、锁信息等数据存储在对象的对象头中。这个过程的具体设置方式取决于 JVM 实现。


6. 执行 init 方法进行初始化

在 Java 程序的视角看来,初始化才正式开始。初始化成员变量,执行实例化代码块,调用类的构造方法,并把堆内对象的首地址赋值给引用变量。


因此一般来说(由字节码中跟随 invokespecial 指令所决定),new 指令之后会接着就是执行构造方法,把对象按照程序员的意愿进行初始化,这样一个真正可用的对象才算完成创建出来。


补充:给对象属性赋值的操作


① 属性的默认初始化(第4步) - ② 显式初始化 / ③ 代码块中初始化 / ④ 构造器中初始化


/**
 *
 *  给对象的属性赋值的操作:
 *  ① 属性的默认初始化 - ② 显式初始化 / ③ 代码块中初始化 - ④ 构造器中初始化
  区别下两类构造器
    <client>类构造器:静态数据初始化 
    <init>实例构造器:非静态数据的初始化
 * @author shkstart  shkstart@126.com
 * @create 2020  17:58
 */
public class Customer{
    int id = 1001;
    String name;
    Account acct;
    {
        name = "匿名客户";
    }
    public Customer(){
        acct = new Account();
    }
}
class Account{
}

显示初始化、代码块初始化、构造器中初始化都在方法中进行

af3821bf8b56eeb189dd1ae6eed9da0d.png

总结

对象实例化的过程

  1. 加载类元信息
  2. 为对象分配内存
  3. 处理并发问题
  4. 属性的默认初始化(零值初始化)
  5. 设置对象头信息
  6. 属性的显示初始化、代码块中初始化、构造器中初始化


8.2. 对象内存布局


4bb4cc3c1b973493c0646d7ed4cd5317.jpg


8.2.1. 对象头(Header)


对象头包含了两部分,分别是运行时元数据(Mark Word)和类型指针。如果是数组,还需要记录数组的长度


运行时元数据

哈希值(HashCode)

GC 分代年龄

锁状态标志

线程持有的锁

偏向线程 ID

翩向时间戳

类型指针

指向类元数据 InstanceKlass,确定该对象所属的类型。


8.2.2. 实例数据(Instance Data)


它是对象真正存储的有效信息,包括程序代码中定义的各种类型的字段(包括从父类继承下来的和本身拥有的字段)


相同宽度的字段总是被分配在一起

父类中定义的变量会出现在子类之前

如果 CompactFields 参数为 true(默认为 true):子类的窄变量可能插入到父类变量的空隙


8.2.3. 对齐填充(Padding)


不是必须的,也没有特别的含义,仅仅起到占位符的作用


举例


public class Customer{
    int id = 1001;
    String name;
    Account acct;
    {
        name = "匿名客户";
    }
    public Customer() {
        acct = new Account();
    }
}
public class CustomerTest{
    public static void main(string[] args){
        Customer cust=new Customer();
    }
}

图示

1bba0ee2a8a614b80d6cac9fecbb3f8a.png

小结

3be25c48b29d3cef51354f19cbe31d7d.png

8.3. 对象的访问定位

375b00ca9022291e7d2e214ce2d7cbfd.png


JVM 是如何通过栈帧中的对象引用访问到其内部的对象实例呢?

deab7773248ea150e5f7dee3901700cd.png

8.3.1. 句柄访问

59cc079fe02b7a5836ff7c2c7fffb635.png

优点: reference 中存储稳定句柄地址,对象被移动(垃圾收集时移动对象很普遍)时只会改变句柄中实例数据指针即可,reference 本身不需要被修改

缺点: 需要专门开辟一片空间用于句柄池,耗费空间多一点。访问时,需要先通过栈中引用找到句柄变量,再从句柄变量访问到对象实例,效率较低。

8.3.2. 直接指针(HotSpot 采用)


694601dcb023c6d10168a00fe000becc.png


直接指针是局部变量表中的引用,直接指向堆中的实例,在对象实例中有类型指针,指向的是方法区中的对象类型数据


优点: 栈中局部变量直指堆中实例,访问效率更高。也不需要专门开辟空间记录句柄。


缺点: 对象被移动时(垃圾收集时移动对象很普遍),reference 会改变。


8.4. 直接内存(Direct Memory)

8.4.1. 直接内存概述


不是虚拟机运行时数据区的一部分,也不是《Java 虚拟机规范》中定义的内存区域。

直接内存是在 Java 堆外的、直接向系统申请的内存区间。

来源于 NIO,通过存在堆中的 DirectByteBuffer 操作 Native 内存。

通常,访问直接内存的速度会优于 Java 堆,即读写性能高。

因此出于性能考虑,读写频繁的场合可能会考虑使用直接内存。

Java 的 NIO 库允许 Java 程序使用直接内存,用于数据缓冲区


8.4.2. 非直接缓存区(传统的IO)

使用 IO 读写文件,需要与磁盘交互,需要由用户态切换到内核态。在内核态时,需要两份内存存储重复数据,效率低。


c691d62d73e977fb942748a81313f5e6.png

8.4.3. 直接缓存区

使用 NIO 时,操作系统划出的直接缓存区可以被 java 代码直接访问,只有一份。NIO 适合对大文件的读写操作。

d13b08447999ab32811a440496e4dac7.png

也可能导致 OutOfMemoryError 异常

Exception in thread "main" java.lang.OutOfMemoryError: Direct buffer memory
    at java.nio.Bits.reserveMemory(Bits.java:693)
    at java.nio.DirectByteBuffer.<init>(DirectByteBuffer.java:123)
    at java.nio.ByteBuffer.allocateDirect(ByteBuffer.java:311)
    at com.atguigu.java.BufferTest2.main(BufferTest2.java:20)

由于直接内存在 Java 堆外,因此它的大小不会直接受限于-Xmx 指定的最大堆大小,但是系统内存是有限的,Java 堆和直接内存的总和依然受限于操作系统能给出的最大内存。


分配回收成本较高

不受 JVM 内存回收管理

直接内存大小可以通过MaxDirectMemorySize设置。如果不指定,默认与堆的最大值-Xmx 参数值一致

0ac6c50a98325c93b5652602137b9dea.png

相关文章
|
3月前
|
缓存 Prometheus 监控
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
643 1
|
2月前
|
存储 Java 程序员
【JVM】——JVM运行机制、类加载机制、内存划分
JVM运行机制,堆栈,程序计数器,元数据区,JVM加载机制,双亲委派模型
|
2月前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
2月前
|
缓存 监控 算法
Python内存管理:掌握对象的生命周期与垃圾回收机制####
本文深入探讨了Python中的内存管理机制,特别是对象的生命周期和垃圾回收过程。通过理解引用计数、标记-清除及分代收集等核心概念,帮助开发者优化程序性能,避免内存泄漏。 ####
64 3
|
2月前
|
缓存 Java
JVM对象引用
本次课程聚焦JVM对象引用,涵盖强引用、软引用、弱引用和虚引用。强引用是最常见的引用类型,确保对象不会被垃圾回收器回收,适用于需要确保对象存活的场景;软引用在内存不足时会被优先回收,常用于缓存;弱引用的对象随时可能被回收,适合临时对象;虚引用最弱,主要用于接收对象回收通知,进行资源清理。通过合理选择引用类型,可优化内存管理,避免内存泄露。
|
3月前
|
Java
JVM内存参数
-Xmx[]:堆空间最大内存 -Xms[]:堆空间最小内存,一般设置成跟堆空间最大内存一样的 -Xmn[]:新生代的最大内存 -xx[use 垃圾回收器名称]:指定垃圾回收器 -xss:设置单个线程栈大小 一般设堆空间为最大可用物理地址的百分之80
|
3月前
|
Java
JVM运行时数据区(内存结构)
1)虚拟机栈:每次调用方法都会在虚拟机栈中产生一个栈帧,每个栈帧中都有方法的参数、局部变量、方法出口等信息,方法执行完毕后释放栈帧 (2)本地方法栈:为native修饰的本地方法提供的空间,在HotSpot中与虚拟机合二为一 (3)程序计数器:保存指令执行的地址,方便线程切回后能继续执行代码
38 3
|
3月前
|
存储 缓存 监控
Elasticsearch集群JVM调优堆外内存
Elasticsearch集群JVM调优堆外内存
73 1
|
4月前
|
存储 安全 Java
jvm 锁的 膨胀过程?锁内存怎么变化的
【10月更文挑战第3天】在Java虚拟机(JVM)中,`synchronized`关键字用于实现同步,确保多个线程在访问共享资源时的一致性和线程安全。JVM对`synchronized`进行了优化,以适应不同的竞争场景,这种优化主要体现在锁的膨胀过程,即从偏向锁到轻量级锁,再到重量级锁的转变。下面我们将详细介绍这一过程以及锁在内存中的变化。
55 4
|
3月前
|
Arthas 监控 Java
JVM进阶调优系列(9)大厂面试官:内存溢出几种?能否现场演示一下?| 面试就那点事
本文介绍了JVM内存溢出(OOM)的四种类型:堆内存、栈内存、元数据区和直接内存溢出。每种类型通过示例代码演示了如何触发OOM,并分析了其原因。文章还提供了如何使用JVM命令工具(如jmap、jhat、GCeasy、Arthas等)分析和定位内存溢出问题的方法。最后,强调了合理设置JVM参数和及时回收内存的重要性。

热门文章

最新文章