基于Java类的加载方式

 更新时间:2023年07月13日 10:13:34   作者:Fluoxetine_Zero  
这篇文章主要介绍了基于Java类的加载方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

类的生命周期

当java源代码文件被javac编译成class文件后,并不能直接运行,而是需要经过加载,连接和初始化这几个阶段后才能使用。

在使用完类或被销毁后,JVM会将类卸载掉。

类加载的过程

类加载的过程需要经过三个阶段分别是:

  • 1.加载
  • 2.连接
  • 3.初始化,其中连接又可分为3个阶段:验证,准备,解析

一、加载(Loading)

由类加载器完成,类的class文件读入内存后,并将其保存到方法区内,然后就会创建一个java.lang.Class类型的对象。

类被载入JVM中,同一个类就不会再次被载入。

需要区分的是“加载”和“类加载”的区别,其中加载只是类加载的第一个环节。

加载阶段:

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

二、验证(Verification)

目的在于确保class文件的字节流中包含的信息符合当前虚拟机的要求,保证被加载类的正确性,不会危害虚拟机自身安全,主要验证包括:

  • 验证文件格式:第一阶段要验证字节流是否符合Class文件格式的规范,并且能被当前版本的虚拟机正确处理。
  • 元数据的验证:第二阶段对字节码描述的信息进行语义分析,比如说验证这个类是不是有父类,类中的字段方法是不是和父类冲突等等,以保证其描述的信息符合Java语言的规范要求。
  • 字节码验证:第三个阶段主要是将对类的方法体(数据流和控制流)进行验证分析。这个阶段保证方法在运行时不会出现危害虚拟机安全的行为语言规范
  • 符号引用验证:第四个阶段符号引用验证可以看做是对类自身以外的信息进行匹配性校验,发生时机是虚拟机将符号引用转换成直接引用时。

三、准备(Preparation)

为类的静态变量(static )分配内存并为其赋零值(默认值0、0.0、false、null等),但是不包含用final修饰的static,因为final在编译时就已经分配了。

不会为实例变量分配初始值,类变量会分配在方法区中,而实例变量是会随着对象一起分配到堆中。

需注意:

  • 这里仅只是给静态变量赋值,而不是成员变量。
  • 在JDK8之前,类的元信息、常量池、静态变量等都存储在永久代这种具体实现中,而在JDK8及以后字符串常量池、静态变量被移除“方法区”,转移到了堆中而元信息,运行时常量池这些依然保留在方法区内,但是具体的存储方式改成了元空间。

四、解析(Resolution)

将常量池中的符号引用替换为直接引用(内存地址)的过程, 主要包括四种类型引用的解析:类或接口的解析、 字段解析、方法解析、接口方法解析。

  • 符号引用:一个Java类被编译成Class之后,如上图,当Test1中引用了Test2,那么在编译阶段,Test1是不知道Test2有没有被编译,也代表Test2一定没有被加载,所以Test1肯定不知道Test2的实际地址。此时在Test1的class的文件中,将使用一个字符串来代表Test2的地址,这个字符串就被称为是符号引用。
  • 直接引用:在运行时,如果Test1发生了类加载,到解析阶段发现Test2还未被加载,这时将会触发Test2的类加载,将Test2加载到虚拟机中,此时Test1中Test2的符号引用将会被替换为Test2的实际地址。

在解析阶段,会将常量池中符号引用替换为直接引用。但是只是替换了部分。这一部分是包含,所有私有方法、静态方法、构造器及初始化方法都是采用静态绑定机制,在编译器阶段就已经指明了调用方法在常量池中的符号引用,JVM运行的时候只需要进行一次常量池解析即可。如果Test1调用的Test2是一个具体的实现类那么就称为静态解析,因为解析的目标类很明确。

那么假如上层Java代码中使用了多态,这里的Test2可能是一个抽象类或者是接口,那么Test2就可能有两个具体的实现类Test3和Test4,这时会因为Test2的具体实现并不明确导致不知道使用哪个具体类的直接引用来进行替换,所以这里就会一直等到运行过程中发生了调用,JVM才会调用栈中将会得到的具体的类型信息,这个时候在进行解析就能用明确的直接引用来替换符号引用,这时解析阶段就会发生在初始化阶段之后,这就是动态解析 用它来实现了后期绑定。

五、 初始化

初始化,则是为标记为常量值的字段赋值的过程。只对static修饰的变量或语句块进行初始化。 如果初始化一个类的时候,其父类尚未初始化,则优先初始化其父类。 如果同时包含多个静态变量和静态初始化块,则按照自上而下的顺序依次执行。

类加载器

Java中默认提供的三种类加载器:

  • 启动类加载器 BootstrapClassLoader(根加载器): 加载Java_Home/jre/lib目录下的核心API
  • 扩展类加载器 ExtClassLoader: 负责加载Java_Home/jre/lib/ext目录下的所有jar包;
  • 应用类加载 AppClassLoader:继承URLClassLoader。对应加载的应用程序classpath目录下的所有jar和class等

双亲委派机制

当一个类加载器收到类加载请求的时候,它首先不会自己去加载这个类的信息,而是把该请求委派给父类加载器,依次向上。

所以所有的类加载请求都会被委派到父类加载器中,只有当父类加载器中无法加载到所需的类,子类加载器才会自己尝试去加载该类。

如果当前类加载器和所有父类加载器都无法加载该类时,则会抛出ClassNotFoundException异常。

双亲委派的作用

1、防止重复加载同一个.class,通过委托确认是否加载,如已加载,无需重复加载,保证数据安全。

2、防止核心.class不能被篡改。

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • Spring--国内Java程序员用得最多的框架

    Spring--国内Java程序员用得最多的框架

    前几年面试最常问的且可以顺利拿到高薪的技能是Spring,随着Spring体系的壮大,除非你在简历上添加Spring Boot和Spring Cloud的技能,才可以打动面试官,而现在,除非是Spring全家桶的实战经验,否则难以让面试官高看
    2021-06-06
  • 基于jmeter实现跨线程组传递token过程图解

    基于jmeter实现跨线程组传递token过程图解

    这篇文章主要介绍了基于jmeter实现跨线程组传递token,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-04-04
  • 利用注解配置Spring容器的方法

    利用注解配置Spring容器的方法

    本篇文章主要介绍了利用注解配置Spring容器的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-12-12
  • Java之打印String对象的地址

    Java之打印String对象的地址

    这篇文章主要介绍了Java之打印String对象的地址,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-09-09
  • Java并发编程之JUC并发核心AQS同步队列原理剖析

    Java并发编程之JUC并发核心AQS同步队列原理剖析

    AbstractQueuedSynchronizer 简称 AQS,可能我们几乎不会直接去使用它,但它却是 JUC 的核心基础组件,支撑着 java 锁和同步器的实现,大神 Doug Lea 在设计 JUC 包时希望能够抽象一个基础且通用的组件以支撑上层模块的实现,AQS 应运而生
    2021-09-09
  • Spring的自定义扩展标签NamespaceHandler解析

    Spring的自定义扩展标签NamespaceHandler解析

    这篇文章主要介绍了Spring的自定义扩展标签NamespaceHandler解析,在很多情况下,我们需要为系统提供可配置化支持,简单的做法可以直接基于Spring的标准Bean来配置,Spring提供了可扩展Schema的支持,这是一个不错的折中方案,需要的朋友可以参考下
    2023-12-12
  • Java中将String类型转换为int类型的几种常见方法

    Java中将String类型转换为int类型的几种常见方法

    在java中经常会遇到需要对数据进行类型转换的场景,这篇文章主要给大家介绍了关于Java中将String类型转换为int类型的几种常见方法,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2024-07-07
  • springboot项目访问图片的3种实现方法(亲测可用)

    springboot项目访问图片的3种实现方法(亲测可用)

    本文主要介绍了springboot项目访问图片的3种实现方法,通过springboot项目访问除项目根目录之外的其它目录的图片,具有一定的参考价值,感兴趣的可以了解一下
    2023-09-09
  • 在Spring环境中正确关闭线程池的姿势

    在Spring环境中正确关闭线程池的姿势

    这篇文章主要介绍了在Spring环境中正确关闭线程池的姿势,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-04-04
  • MyBatis-plus与数据库字段的映射方式

    MyBatis-plus与数据库字段的映射方式

    MyBatis-plus默认开启驼峰命名规则映射,即驼峰转为下划线,如果数据库字段也是驼峰命名,会导致映射错误,解决方法有两种:使用@TableField注解指定数据库字段名或关闭驼峰命名规则映射,数据库设计规范应优先使用蛇形命名法,即全部小写字母并用下划线连接
    2025-11-11

最新评论