一些收藏

2. new背后发生了什么-类加载

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

概述

new一个对象包括:类加载(未加载时)、为对象分配内存、内存空间初始化、对象头设置等步骤。

1. 如何判断类是否被加载

--示例代码
public class Demo {
    public void tests(String a){
        User user = new User();
    }

    public class User {
        String name;
    }
}
tests方法本地变量表
字节码常量池

2. 类生命周期

类生命周期.png

3. 加载

3.1 通过类的全限定名在类加载器路径中获取此类的二进制字节流
3.2 将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构
例如:
Classfile /D:/privatefile/LeetCode/src/zzzliu/JVM/Demo.class
  Last modified 2020-11-15; size 559 bytes
  MD5 checksum a962546105384ba27fde2199fdba1a74
  Compiled from "Demo.java"
public class zzzliu.JVM.Demo
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Methodref          #5.#25         // java/lang/Object."<init>":()V
   #2 = Class              #26            // zzzliu/JVM/Demo$User
   #3 = Methodref          #2.#27         // zzzliu/JVM/Demo$User."<init>":(Lzzzliu/JVM/Demo;)V
   #4 = Class              #28            // zzzliu/JVM/Demo
   #5 = Class              #29            // java/lang/Object
   #6 = Utf8               User
   #7 = Utf8               InnerClasses
   #8 = Utf8               <init>
   #9 = Utf8               ()V
  #10 = Utf8               Code
  #11 = Utf8               LineNumberTable
  #12 = Utf8               LocalVariableTable
  #13 = Utf8               this
  #14 = Utf8               Lzzzliu/JVM/Demo;
  #15 = Utf8               tests
  #16 = Utf8               (Ljava/lang/String;)V
  #17 = Utf8               a
  #18 = Utf8               Ljava/lang/String;
  #19 = Utf8               b
  #20 = Utf8               I
  #21 = Utf8               user
  #22 = Utf8               Lzzzliu/JVM/Demo$User;
  #23 = Utf8               SourceFile
  #24 = Utf8               Demo.java
  #25 = NameAndType        #8:#9          // "<init>":()V
  #26 = Utf8               zzzliu/JVM/Demo$User
  #27 = NameAndType        #8:#30         // "<init>":(Lzzzliu/JVM/Demo;)V
  #28 = Utf8               zzzliu/JVM/Demo
  #29 = Utf8               java/lang/Object
  #30 = Utf8               (Lzzzliu/JVM/Demo;)V
{
  public zzzliu.JVM.Demo();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      LineNumberTable:
        line 3: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   Lzzzliu/JVM/Demo;

  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/Demo$User
         5: dup
         6: aload_0
         7: invokespecial #3                  // Method zzzliu/JVM/Demo$User."<init>":(Lzzzliu/JVM/Demo;)V
        10: astore_3
        11: return
      LineNumberTable:
        line 6: 0
        line 8: 2
        line 9: 11
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      12     0  this   Lzzzliu/JVM/Demo;
            0      12     1     a   Ljava/lang/String;
            2      10     2     b   I
           11       1     3  user   Lzzzliu/JVM/Demo$User;
}
SourceFile: "Demo.java"
InnerClasses:
     public #6= #2 of #4; //User=class zzzliu/JVM/Demo$User of class zzzliu/JVM/Demo
3.3 在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口
class对象

4. 验证

5. 准备

--示例代码
private static final int value = 100;

--编译时javac就会为value生成ConstantValue属性
private static final int value;
    descriptor: I
    flags: ACC_PRIVATE, ACC_STATIC, ACC_FINAL
    ConstantValue: int 100

6. 解析

6.1 类或接口解析
--常量池中类的符号引用
#38 = Utf8               zzzliu/JVM/Demo
#39 = Utf8               java/lang/Object
  1. 将符号应用传递给当前类的类加载器去加载该符号引用的类;加载过程中由于元数据验证、字节码验证的需要,有可能触发父类或接口的加载;
  2. 第一次解析后结果缓存在常量池中,之后直接获取;
  3. 解析完成之前对符号引用进行验证,确认是否具备该类的访问权限;
6.1 字段解析
private User user;

private zzzliu.JVM.Demo$User user;
    descriptor: Lzzzliu/JVM/Demo$User;
    flags: ACC_PRIVATE

先解析字段所属类或接口为C,然后按如下顺序解析字段

  1. C本身就包含了名称和字段描述都与目标匹配的字段,则返回这个字段的直接引用;
  2. 否则,如果C继承了接口,按照从下往上的继承关系往上搜索,查找到则直接返回;
  3. 否则,按照C的继承关系往上搜索,查找到则直接返回;
  4. 否则,都未查找到则抛出java.lang.NoSuchFieldError
6.1 方法解析

同字段解析类似,先解析方法所属类或接口为C,然后按如下顺序解析字段

  1. C本身就包含了名称和字段描述都与目标匹配的方法,则返回这个方法的直接引用;
  2. 否则,按照C的继承关系往上搜索,查找到则直接返回;
  3. 否则,如果C继承了接口,按照从下往上的继承关系往上搜索,查找到则说明C是个接口,抛出java.lang.AbstractMethodError
  4. 否则,都未查找到则抛出java.lang.NoSuchMethodError

最后查找过程成功后检查权限,如果不具备访问权限,抛出java.lang.IllegalAccessError

7. 初始化

初始化阶段就是执行类构造器<clinit>()方法的过程,<clinit>()是Javac编译器自动生成的;

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

上一篇 下一篇

猜你喜欢

热点阅读