Thinking in Java

3. new背后发生了什么-对象创建

2020-11-22  本文已影响0人  进击的蚂蚁zzzliu

概述

上一节分析了类加载时会把class文件中静态数据结构(包括常量池、方法表(方法字节码指令)、字段表等)转化为方法区的运行时数据结构,本节继续分析new背后的后续流程;本节主要以下面两个类为例进行分析

public class User {
    public static int money = 99;
    private String name;
    private int age = 3;
    public User(String name) {
        this.name = name;
    }
}
public class Demo {
    public void tests(String a){
        int temp = 1;
        User user = new User("zzzliu");
    }
}
运行时数据模型.png

1. 'new'指令执行

--tests方法字节码反汇编信息:
public void tests(java.lang.String);
    descriptor: (Ljava/lang/String;)V
    flags: ACC_PUBLIC
    Code:
      stack=3, locals=4, args_size=2
         0: iconst_1
         1: istore_2
         2: new           #2                  // class zzzliu/JVM/User
         5: dup
         6: ldc           #3                  // String zzzliu
         8: invokespecial #4                  // Method zzzliu/JVM/User."<init>":(Ljava/lang/String;)V
        11: astore_3
        12: return
      LineNumberTable:
        line 6: 0
        line 7: 2
        line 8: 12
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      13     0  this   Lzzzliu/JVM/Demo;
            0      13     1     a   Ljava/lang/String;
            2      11     2  temp   I
           12       1     3  user   Lzzzliu/JVM/User;

注意:invokespecial指令(执行构造方法初始化实例变量)和astore指令(将user指向分配的内存空间,此时user已经不为null)有可能发生指令重排序,导致多线程时其他线程有可能拿到不为null但是实例变量尚未赋值的对象; 这也就是DCL单例时为什么要用volatile防止指令重排序的原因

2. 内存分配

内存分配主要涉及三个问题 1. 需要分配多大内存?2. 在哪儿分配?3. 如何分配

2.1 对象占用内存大小计算

对象组成.png

涉及到的压缩指针、字段重排列、虚共享问题后期单独分析

2.2 在哪儿分配?

对象首先尝试在TLAB进行分配,无法分配时再尝试在共享堆中分配

具体分配在Eden区、Survivor-From区、Survivor-From区、老年代、还是Region本节先不分析

2.3 如何分配?

由于JVM使用垃圾收集器不同决定了使用不同的GC算法,不同的GC算法决定了不同的内存形态,不同的内存形态又决定了不同的分配方式

3. 初始化零值

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

注意:JVM不会给局部变量赋零值,因此局部变量不显示赋值编译器会报错

4. 设置对象头

  1. MarkWord: 64位JVM占用8字节
  2. MetaData: 指向方法区class对象的指针,开启压缩指针占4字节未开启占8字节(Hotspot默认开启)
  3. ArrayLength: 数组长度,数组类型才有,占用4字节
64位Hotspot对象头MarkWord

5. 执行<init>方法

至此从虚拟机的视角来看,一个新的对象已经产生了,但从Java程序的视角来看,所有的字段都还为零,无法使用。需要执行<init>方法初始化后,这样一个真正可用的对象才算完全创建出来;

5.1 <init>字节码指令执行

public static int money;
  descriptor: I
  flags: ACC_PUBLIC, ACC_STATIC

private java.lang.String name;
  descriptor: Ljava/lang/String;
  flags: ACC_PRIVATE

private int age;
  descriptor: I
  flags: ACC_PRIVATE

--User <init>方法
public zzzliu.JVM.User(java.lang.String);
  descriptor: (Ljava/lang/String;)V
  flags: ACC_PUBLIC
  Code:
    stack=2, locals=2, args_size=2
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: aload_0
       5: iconst_3
       6: putfield      #2                  // Field age:I
       9: aload_0
      10: aload_1
      11: putfield      #3                  // Field name:Ljava/lang/String;
      14: return
    LocalVariableTable:
      Start  Length  Slot  Name   Signature
          0      15     0  this   Lzzzliu/JVM/User;
          0      15     1  name   Ljava/lang/String;

5.2 <init>和<clinit>区别

--clinit方法字节码指令:
static {};
    descriptor: ()V
    flags: ACC_STATIC
    Code:
      stack=1, locals=0, args_size=0
         0: bipush        99
         2: putstatic     #4                  // Field money:I
         5: return
      LineNumberTable:
        line 4: 0

总结

new对象主要分5个步骤:类加载 - 分配内存 - 初始化零值 - 设置对象头 - 执行构造方法

--------over---------

上一篇 下一篇

猜你喜欢

热点阅读