jvm 类的加载流程详解

1 概述

java中数据类型分数为基本数据类型和引用数据类型,基本数据类型由虚拟机预先定义,引用数据类型则需要进行类的加载。

2 loading 加载

2.1 加载概述

加载就是将java类的字节码文件加载到机器内存中,在内存中构建java类的原型——类模板对象。所谓的类模板对象,其实就是java类在jvm内存中的一个块照,jvm将字节码文件中解析出的常量池,类字段,类方法等信息存储到类模板中,jvm在运行期就能通过类模板获取java类中任意信息。(反射就是基于此基础)

2.2 加载流程

查找并加载类的二进制数据,生成class的实例

  • 通过类的全名,获取类的二进制数据流

  • 解析类的二进制数据流为方法区的数据结构(java类模型)

  • 创建java.lang.Class 类实例表示该类型

2.2.1 二进制流的获取方式

只要所读取的字节码符合jvm规范即可

  • 虚拟机读取class后缀的文件(常见)

  • 读入jar,zip等数据包

  • 使用http之类的协议通过网络加载

在获取到类的二进制信息后,jvm处理这些数据,并最终产生java.lang.class的实例

2.2.2 类模型及class类实例的位置
  • 类模型位置
    加载的类在jvm中创建相应的类结构,类结构存储在方法区(jdk1.8之前:永久代,jdk1.8之后 元空间)

  • class实例位置
    类将class文件加载到元空间后,会在堆中创建一个java.lang.class对象,用于封装方法区的数据结构,此过程在类加载过程中创建。

3 linking 连接阶段

3.1 验证

验证阶段包括:格式验证,语义检查,字节码验证,以及符号引用验证

  • 其中格式验证会和加载阶段一起执行

  • 格式验证之外的验证操作会在方法区中进行

3.2 准备阶段

为类的静态变量分配内存,为其初始化为默认值

注意:这里不包括基本数据类型字段用static final修饰的情况,因为final在编译的时候就会分配了,准备阶段会显式赋值。

3.3 解析阶段

解析阶段是将类,接口,字段和方法的符号引用转为直接引用。
解析操作往往会在jvm执行完初始化之后在执行。

如果使用符号引用,虚拟机其实也不知道具体引用的类的内存地址,那么也就无法真正的调用到该类,所以要把符号引用转为直接引用,这样就能够真正定位到类在内存中的地址,如果符号引用转直接引用失败,就说明类还没有被加载到内存中,就会报错

4 初始化阶段

为类的静态变量赋予正确的初始值

4.1 流程

到了初始化阶段,此时类才会执行Java字节码(只有到了初始化,才会执行类中定义的java程序)重要的工作时执行类的初始化方法:()方法

  • 由静态成员的赋值以及static语句块合并产生

  • 该方法仅有java编译器生成,并由jvm进行调用。

注意:在加载一个类之前,虚拟机会试图加载该类的父类,因此父类的总是在子类之前被调用,也就是说父类的static块优先于子类

结论: 在连接阶段的准备环节赋值情况:

1.对于基本数据类型的字段,static final 修饰(直接赋值常量,而不是方法调用) 则是在连接阶段

2.对于string来说,如果是字面量方式赋值+static final修饰,则也是在连接阶段

3.除了上述的以为的都在初始化阶段赋值

5 类的主动使用与被动使用

  • 主动使用(类会被初始化执行)

  • 被动使用(不会执行初始化,但会被加载)

class只有在必须首次的使用时才会被装载

5.1 主动使用

  • 创建一个类的实例,比如:new,反射,序列化

  • 调用类的静态方法

  • 使用类,接口的静态字段时

  • 使用 java.lang.reflect 包中的方法反射类的方法时

  • 初始化子类时,若发现父类未初始化,则先进行父类初始化
    注意:不会初始化他所实现的接口,当程序首次使用接口的特定字段时才会初始化。

  • 如果接口定义了default方法,则接口在会被在所实现类之间进行初始化

  • 虚拟机启动时,用户指定要执行的主类(含main方法),虚拟机会初始化这个类

  • 当调用methodHandle实例时,初始化该methodHandle指向的方法所在的类

5.2 被动使用

被动使用不会引起类的初始化。并不是在代码中出现的类,就一定会被 加载或者初始化。如果不符合主动使用的条件,类就不会初始化。

  • 当访问一个静态字段时,只有真正声明这个字段的类才会被初始化。

    • 当通过子类引用父类的静态变量时,不会导致子类初始化
  • 通过数组定义类引用

  • 引用常量不会触发此类或接口的初始化

  • 调用classloader类的loadclass方法加载一个类

注意:没有初始化的一个类不意味着没有被加载

6 类的使用

类被加载完后就可以使用类的属性 或者创建对象了

7类的卸载

7.1 类,类加载器,类实例之间关系

通过类加载器可以加载class对象,另一方面 一个class对象总是会引用他的类加载器,调用class对象的getClassloader()方法,会获得它的类加载器。

一个类的实例总是引用代表这个类的class对象 ,在object中定义了getclass()方法,代表所属类class对象的引用,此外java类都有静态属性class,它引用代表这个类的class对象。

7.2 类的生命周期

当sqmple类的class对象不再被引用时,即不可触及时,class对象就会结束生命周期,sample类在方法区内数据也会被卸载

一个类何时结束生命周期,取决于代表它的class对象何时结束生名周期

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值