Java类加载过程详解


首先我们先简单了解一下整个JVM的体系结构是怎样的,如下图所示(图1):
在这里插入图片描述
再详细具体一点(图2):
在这里插入图片描述
上面的图片有些地方看不懂没关系,先在脑子里大体有个印象。
我们这篇文章讲的是类的加载过程,所以在讲类加载之前先要知道类的生命周期是一个怎样的过程。
如下图所示(图3):
在这里插入图片描述

类的生命周期一共有5个阶段,分别是:
1.加载(Loading)
2.链接(Linking)
3.初始化(Initialization)
4.使用(Using)
5.卸载(UNLoading)

其中链接(Linking)阶段又可以细分为三个阶段:
1)验证(Verification)
2)准备(Preparation)
3)解析(Resolution)

今天我们只讲类的加载过程。如下图所示(图4):
在这里插入图片描述

类的加载

类的加载:指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构。
类的加载的最终产品是位于堆区中的Class对象,Class对象封装了类在方法区内的数据结构,并且向Java程序员提供了访问方法区内的数据结构的接口。

类加载过程

通过以上图片我们可以看出类的加载过程是发生在类装载器子系统中的。

类装载器子系统:虚拟机把描述类的数据从class文件加载到内存,并对数据进行验证、准备、解析和初始化,最终形成可以被虚拟机直接使用的Java类型。

加载

加载是类加载过程的第一个阶段,不要将这2个概念混淆了。

1)通过一个类的全限定名来获取其定义的二进制字节流。
2)将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构。
3)在Java堆中生成一个代表这个类的java.lang.Class对象,作为对方法区中这些数据的访问入口。

链接

验证

确保被加载的类的正确性。

验证是连接阶段的第一步,这一阶段的目的是为了确保Class文件的字节流中包含的信息符合

当前虚拟机的要求,并且不会危害虚拟机自身的安全。

  • 文件格式验证:验证字节流是否符合Class文件格式的规范,如:是否以模数0xCAFEBABE开头、主次版本号是否在当前虚拟机处理范围内等等。
  • 元数据验证:对字节码描述的信息进行语义分析,以保证其描述的信息符合Java语言规范的要求;如:这个类是否有父类,是否实现了父类的抽象方法,是否重写了父类的final方法,是否继承了被final修饰的类等等。
  • 字节码验证:通过数据流和控制流分析,确定程序语义是合法的、符合逻辑的,如:操作数栈的数据类型与指令代码序列能配合工作,保证方法中的类型转换有效等等。
  • 符号引用验证:确保解析动作能正确执行;如:通过符合引用能找到对应的类和方法,符号引用中类、属性、方法的访问性是否能被当前类访问等等。

验证阶段是非常重要的,但不是必须的,它对程序运行期没有影响,如果所引用的类经过反

复验证,那 么可以考虑采用-Xverifynone参数来关闭大部分的类验证措施,以缩短虚拟机类

加载的时间。

准备

为静态变量(类变量)分配内存并且设置该静态变量(类变量)的默认初始值,即零值。

这些内存都将在方法区中进行分配。

  • 只对static修饰的静态变量进行内存分配、赋默认值(如0、0L、null、false等)。
  • 对final的静态字面值常量直接赋初值(赋初值不是赋默认值,如果不是字面值静态常量,那么会和静态变量一样赋默认值)。

解析

将虚拟机常量池内的符号引用解析为直接引用,指到内存中的具体地址。

  • 解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程,解析动作主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用限定符7类符号引用进行。符号引用就是一组符号来描述目标,可以是任何字面量。
    直接引用就是直接指向目标的指针、相对偏移量或一个间接定位到目标的句柄。

初始化

为类的静态变量赋初值。

初始化阶段是执行类构造器<clinit>()方法的过程。

类构造器<clinit>()方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块

(static块)中的语句合并产生的。也就是说,当我们代码中包含static变量的时候,就会有

<clinit>( )方法;如果当前类不存在static变量,那么它的字节码文件是不会存在<clinit>( )

  • <clinit>( )方法中的指令按语句在源文件中出现的顺序执行
  • <clinit>( )不同于类的构造器(构造方法)。(关联:构造器是虚拟机视角下的<init>( ))
  • 若该类具有父类,JVM会保证子类的<clinit>( )执行前,父类的<clinit>( )已经执行完毕
  • 虚拟机会保证一个类的<clinit>( )方法在多线程环境中被正确加锁和同步。

注意:需要明确类加载过程类初始化类的实例化区别,具体内容等再写一篇文章。

总结

最后用一张图总结这个过程。
在这里插入图片描述

  • 3
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 6
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值