类的生命周期

110 阅读6分钟

类的生命周期可以分为五个阶段:加载连接初始化使用卸载

  • 加载阶段
    • 第一步:类加载器根据类的全限定名通过不同的渠道以二进制流的方式获取字节码信息
    • 第二步:类加载器在加载完类之后,Java虚拟机会将字节码中的信息保存到方法区中
    • 第三步:生成一个IstanceKlass对象,保存类的所有信息,里边还包含实现特定功能比如多态的信息
    • 第四步:同时,Java虚拟机还会在堆中生成一份与方法去中数据类似的java.lang.Class对象

作用是在Java代码中去获取类的信息以及存储静态字段的数据(JDK8及之后)

程序员一般用到java.lang.Class对象

  • 连接阶段 连接阶段可以分为三个阶段验证准备解析

- 连接阶段的第一个环节是验证,验证的主要目的是检测Java字节码文件是否遵守了《Java虚拟机规范》中的约束,这个阶段一般不需要程序员参与
    * 文件格式校验,比如字节码文件是否以0xCAFEBABE开头,主次版本号是否满足当前Java虚拟机版本的要求
    * 元信息验证,例如类必须有父类(super不能为空,默认为java.lang.Object)
    * 验证程序执行指令的语义,比如方法内的指令执行中跳转到其它方法中去
    * 符号引用验证,例如是否访问了其他类中private的方法等
- 准备阶段为静态变量(**static**)分配内存并设置初始值(默认值)**final**修饰的**基本数据类型的静态变量**,准备阶段直接会将代码中的值进行赋值

- 解析阶段主要是将常量池中的符号引用替换为直接引用
- `符号引用`就是在字节码文件中使用编号访问常量池中的内容
- `直接引用`不再使用编号,而是使用内存中地址进行访问具体的数据
- 直接引用:![](https://cdn.nlark.com/yuque/0/2024/png/38471228/1729606625409-0ed591e7-453e-40e9-a3e3-6afdce3ed25b.png)
  • **初始化阶段 **
    • 会执行静态代码块中的代码,并为静态变量赋值
    • 会执行字节码文件中clinit部分的字节码指令,clinit是给静态变量赋值
    • clinit方法中的执行顺序与Java中编写顺序是一致的

jvm可以添加-XX+TraceClassLoading参数可以打印出加载并初始化的类

以下几种方式会导致类的初始化:

    * 访问一个类的静态变量或者静态方法,变量是final修饰并且等号是右边是常量不会触发初始化
    * 调用Class.forName(String className)
    * new一个该类的对象时
    * 执行Main方法的当前类
- 字节码文件没有**clinit**的几种情况(不执行初始化指令)
    * 无静态代码块且无静态变量赋值语句
    * 有静态变量的声明,但是没有赋值语句
    * 静态变量的定义使用final关键字,这类变量会在准备阶段直接进行初始化
    * 直接访问**父类**的静态变量,不会触发子类的初始化
    * 子类的初始化clinit调用之前,会先调用父类的clinit初始化方法![](https://cdn.nlark.com/yuque/0/2024/png/38471228/1729654575412-2e08aa12-9c9e-4a03-8c48-bddea15522c5.png)
    * 数组的创建不会导致数组中元素的类进行初始化

类的生命周期可以分为五个阶段:加载连接初始化使用卸载

  • 加载阶段
    • 第一步:类加载器根据类的全限定名通过不同的渠道以二进制流的方式获取字节码信息
    • 第二步:类加载器在加载完类之后,Java虚拟机会将字节码中的信息保存到方法区中
    • 第三步:生成一个IstanceKlass对象,保存类的所有信息,里边还包含实现特定功能比如多态的信息
    • 第四步:同时,Java虚拟机还会在堆中生成一份与方法去中数据类似的java.lang.Class对象

作用是在Java代码中去获取类的信息以及存储静态字段的数据(JDK8及之后)

程序员一般用到java.lang.Class对象

  • 连接阶段 连接阶段可以分为三个阶段验证准备解析

- 连接阶段的第一个环节是验证,验证的主要目的是检测Java字节码文件是否遵守了《Java虚拟机规范》中的约束,这个阶段一般不需要程序员参与
    * 文件格式校验,比如字节码文件是否以0xCAFEBABE开头,主次版本号是否满足当前Java虚拟机版本的要求
    * 元信息验证,例如类必须有父类(super不能为空,默认为java.lang.Object)
    * 验证程序执行指令的语义,比如方法内的指令执行中跳转到其它方法中去
    * 符号引用验证,例如是否访问了其他类中private的方法等
- 准备阶段为静态变量(**static**)分配内存并设置初始值(默认值)**final**修饰的**基本数据类型的静态变量**,准备阶段直接会将代码中的值进行赋值

- 解析阶段主要是将常量池中的符号引用替换为直接引用
- `符号引用`就是在字节码文件中使用编号访问常量池中的内容
- `直接引用`不再使用编号,而是使用内存中地址进行访问具体的数据
- 直接引用:![](https://cdn.nlark.com/yuque/0/2024/png/38471228/1729606625409-0ed591e7-453e-40e9-a3e3-6afdce3ed25b.png)
  • **初始化阶段 **
    • 会执行静态代码块中的代码,并为静态变量赋值
    • 会执行字节码文件中clinit部分的字节码指令,clinit是给静态变量赋值
    • clinit方法中的执行顺序与Java中编写顺序是一致的

jvm可以添加-XX+TraceClassLoading参数可以打印出加载并初始化的类

以下几种方式会导致类的初始化:

    * 访问一个类的静态变量或者静态方法,变量是final修饰并且等号是右边是常量不会触发初始化
    * 调用Class.forName(String className)
    * new一个该类的对象时
    * 执行Main方法的当前类
- 字节码文件没有**clinit**的几种情况(不执行初始化指令)
    * 无静态代码块且无静态变量赋值语句
    * 有静态变量的声明,但是没有赋值语句
    * 静态变量的定义使用final关键字,这类变量会在准备阶段直接进行初始化
    * 直接访问**父类**的静态变量,不会触发子类的初始化
    * 子类的初始化clinit调用之前,会先调用父类的clinit初始化方法![](https://cdn.nlark.com/yuque/0/2024/png/38471228/1729654575412-2e08aa12-9c9e-4a03-8c48-bddea15522c5.png)
    * 数组的创建不会导致数组中元素的类进行初始化