基于Java类的加载方式

2023-08-06 0 5,062

目录

类的生命周期

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

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

基于Java类的加载方式

类加载的过程

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

  • 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类的加载方式

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

  • 符号引用:一个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等

双亲委派机制

基于Java类的加载方式

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

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

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

双亲委派的作用

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

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

总结

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

资源下载此资源下载价格为1小猪币,终身VIP免费,请先
由于本站资源来源于互联网,以研究交流为目的,所有仅供大家参考、学习,不存在任何商业目的与商业用途,如资源存在BUG以及其他任何问题,请自行解决,本站不提供技术服务! 由于资源为虚拟可复制性,下载后不予退积分和退款,谢谢您的支持!如遇到失效或错误的下载链接请联系客服QQ:442469558

:本文采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可, 转载请附上原文出处链接。
1、本站提供的源码不保证资源的完整性以及安全性,不附带任何技术服务!
2、本站提供的模板、软件工具等其他资源,均不包含技术服务,请大家谅解!
3、本站提供的资源仅供下载者参考学习,请勿用于任何商业用途,请24小时内删除!
4、如需商用,请购买正版,由于未及时购买正版发生的侵权行为,与本站无关。
5、本站部分资源存放于百度网盘或其他网盘中,请提前注册好百度网盘账号,下载安装百度网盘客户端或其他网盘客户端进行下载;
6、本站部分资源文件是经压缩后的,请下载后安装解压软件,推荐使用WinRAR和7-Zip解压软件。
7、如果本站提供的资源侵犯到了您的权益,请邮件联系: 442469558@qq.com 进行处理!

猪小侠源码-最新源码下载平台 Java教程 基于Java类的加载方式 http://www.20zxx.cn/806263/xuexijiaocheng/javajc.html

猪小侠源码,优质资源分享网

常见问题
  • 本站所有资源版权均属于原作者所有,均只能用于参考学习,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担
查看详情
  • 最常见的情况是下载不完整: 可对比下载完压缩包的与网盘上的容量,建议提前注册好百度网盘账号,使用百度网盘客户端下载
查看详情

相关文章

官方客服团队

为您解决烦忧 - 24小时在线 专业服务