【JVM】运行时数据区与对象的创建流程

文章详细介绍了JVM的运行时数据区,包括堆、元空间、线程栈、本地方法区和程序计数器。在方法执行时,线程栈会创建栈帧,包含局部变量表、操作数栈等。对象创建过程涉及类加载校验、内存分配(如指针碰撞、空闲列表、TLAB)、设置初始值和对象头信息,包括对象哈希码、锁状态等。文章还讨论了指针压缩技术,以减少大内存环境下的带宽占用和GC压力,并提供了示例代码展示对象指针压缩的效果。
摘要由CSDN通过智能技术生成

4、运行时数据区

4.1、运行时数据区介绍

在这里插入图片描述

运行时数据区也就是JVM在运⾏时产生的数据存放的区域,这块区域就是JVM的内存区域,也称为JVM的内存模型——JMM

  • 堆空间(线程共享):存放new出来的对象

  • 元空间(线程共享):存放类元信息、类的模版、常量池、静态部分

  • 线程栈(线程独享):⽅法的栈帧

  • 本地⽅法区(线程独享):本地⽅法产⽣的数据

  • 程序计数器(线程独享):配合执⾏引擎来执⾏指令

4.2、程序在执行时运行时数据区中的内存变化

⾸先,在程序的.class目录内执行如下命令,查看程序具体的汇编指令

javap -c JVMAnalyze

得到结果:

Compiled from "JVMAnalyze.java"
public class com.qf.jvm.JVMAnalyze {
  public com.qf.jvm.JVMAnalyze();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return

  public int add();
    Code:
       0: bipush        10
       2: istore_1
       3: bipush        20
       5: istore_2
       6: iload_1
       7: iload_2
       8: iadd
       9: bipush        10
      11: imul
      12: istore_3
      13: iload_3
      14: ireturn

  public static void main(java.lang.String[]);
    Code:
       0: new           #2                  // class com/qf/jvm/JVMAnalyze
       3: dup
       4: invokespecial #3                  // Method "<init>":()V
       7: astore_1
       8: aload_1
       9: invokevirtual #4                  // Method add:()I
      12: istore_2
      13: getstatic     #5                  // Field java/lang/System.out:Ljava/io/PrintStream;
      16: iload_2
      17: invokevirtual #6                  // Method java/io/PrintStream.println:(I)V
      20: return
}

通过指令,JMM内存发⽣了如下变化:

在这里插入图片描述

  • 线程栈:执行⼀个方法就会在线程栈中创建⼀个栈帧。

  • 栈帧包含如下四个内容:

    • 局部变量表:存放方法中的局部变量
    • 操作数栈:用来存放方法中要操作的数据
    • 动态链接:存放方法名和方法内容的映射关系,通过方法名找到方法内容
    • 方法出口:记录方法执行完后调用此方法的位置。

5、对象的创建流程

5.1、对象创建流程

在这里插入图片描述

5.2、类加载校验

校验该类是否已被加载。主要是检查常量池中是否存在该类的类元信息。如果没有,则需要进⾏加载。

5.3、分配内存

为对象分配内存。具体的分配策略如下:

  • Bump the Pointer(指针碰撞):如果内存空间的分配是绝对规整的,则JVM记录当前剩余内存的指针,在已用内存分配

  • Free List(空闲列表):如果内存空间的分配不规整,那么JVM会维护⼀个可用内存空间的列表用于分配。

对象并发分配存在的问题:

  • Compare And Swap:自旋分配,如果并发分配失败则重试分配之后的地址

  • Thread Local Allocation Buffer(TLAB):本地线程分配缓冲,JVM为每个线程分配⼀块空间,每个线程在自己的空间中创建对象(jdk8默认使⽤,之前版本需要通过-XX:+UseTLAB开启)

5.4、设置初值

根据数据类型,为对象空间赋初始化值。

5.5、设置对象头

为对象设置对象头信息,对象头信息包含以下内容:类元信息、对象哈希码、对象年龄、锁状态标志等。

  • 对象头中的Mark Work 字段(32位)

在这里插入图片描述

  • 对象头中的类型指针(Klass Pointer)

类型指针用于指向元空间当前类的类元信息。比如调用类中的方法,通过类型指针找到元空间中的该类,再找到相应的方法。

开启指针压缩后,类型指针只用4个字节存储,否则需要8个字节存储

  • 指针压缩

过大的对象地址,会占⽤更大的带宽和增加GC的压力。

对象中指向其他对象所使⽤的指针:8字节被压缩成4字节。 最早的机器是32位,最大支持内存 2的32次方=4G。现在是64位,2的64次⽅可以表示N个T的内存。内存32G即等于2的35次方。如果内存是32G的话,用35位表示内存地址,这样过于浪费。如果把35位的数据,根据算法,压缩成32位的数据(也就是4个字节)。在保存时用4个字节,再使用时使用8个字节。之前用35位保存内存地址,就可以用32位保存。这样8个字节的对象,实际上使用32位来保存,这样64位就能表示2个对象。

如果内存⼤于32G,指针压缩会失效,会强制使用64位来表示对象地址。因此jvm堆内存最好不要大于32G。

Jdk1.6之后默认开启指针压缩,可通过配置jvm参数关闭指针要锁 -XX:-UseCompressedOops

示例代码:

package com.qf.jvm;

import org.openjdk.jol.info.ClassLayout;

import java.lang.String;

/**
 * 对象指针压缩
 * @author Thor
 * @公众号 Java架构栈
 */
public class ObjectLengthAnalyze {

    public static void main(String[] args) {
        ClassLayout classLayout = ClassLayout.parseInstance(new A());
        System.out.println(classLayout.toPrintable());
    }

    static class A{
        int num;
        String name;
    }
}

关闭指针压缩:

在这里插入图片描述

开启指针压缩:

在这里插入图片描述

5.6、执行init方法

为对象中的属性赋值和执⾏构造方法。

本文章参考B站 千锋教育JVM全套教程(含jvm调优、jvm虚拟机、jvm面试题、jvm源码详解)系统玩转java虚拟机全程干货无废话,仅供个人学习使用,部分内容为本人自己见解,与千锋教育无关。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值