当前位置: 首页 > article >正文

面试小札:Java的类加载过程和类加载机制。

Java类加载过程

 

加载(Loading)

这是类加载过程的第一个阶段。在这个阶段,Java虚拟机(JVM)主要完成三件事:

通过类的全限定名来获取定义此类的二进制字节流。这可以从多种来源获取,如本地文件系统(.class文件)、网络(如从远程服务器下载字节码)、动态生成字节码(如使用字节码生成库)等。

将字节流所代表的静态存储结构转换为方法区(在JDK 1.8之后,元数据存储在本地内存的元空间Metaspace中)中的运行时数据结构。

在内存中生成一个代表这个类的 java.lang.Class 对象,作为方法区这个类的各种数据的访问入口。这个 Class 对象在堆内存中,它就像是一面镜子,反射出类在方法区中的结构。

验证(Verification)

目的是确保被加载的类的字节码是合法的、符合Java虚拟机规范的。它主要包括四个验证阶段:

文件格式验证:验证字节流是否符合Class文件格式的规范,例如是否以魔数( 0xCAFEBABE )开头,主次版本号是否在当前JVM支持的范围内等。

元数据验证:对字节码描述的信息进行语义分析,以保证其符合Java语言规范,例如检查这个类是否有父类(除了 java.lang.Object ),是否继承了不允许继承的类(如 final 类)等。

字节码验证:通过数据流和控制流分析,确定程序语义是合法的、符合逻辑的。例如检查操作数栈的数据类型是否和指令的操作码相匹配,跳转指令是否会跳转到方法体以外的字节码指令上。

符号引用验证:在解析阶段将符号引用转换为直接引用的时候,对符号引用进行验证。这个阶段主要是确保解析行为能正常执行,比如检查符号引用中的类、字段、方法是否确实存在等。

准备(Preparation)

这一阶段是为类的静态变量(被 static 修饰的变量)分配内存并设置默认初始值。例如对于 public static int value = 123; ,在准备阶段, value 会被初始化为0(基本数据类型的默认值),而不是123。这里需要注意的是,这一阶段不会执行任何Java代码,仅仅是为变量分配内存和设置默认值。

解析(Resolution)

这是虚拟机将常量池内的符号引用替换为直接引用的过程。符号引用是一种对目标的描述,例如一个类的全限定名、方法的名称和描述符等。直接引用是指向目标的指针、相对偏移量或者能间接定位到目标的句柄。例如,在调用一个方法时,需要将方法的符号引用解析为实际内存中的方法入口地址(直接引用)。这个过程主要针对类或接口、字段、类方法、接口方法等符号引用进行解析。

初始化(Initialization)

这是类加载过程的最后一步,也是真正开始执行类中定义的Java程序代码的阶段。这个阶段主要是执行类构造器 <clinit>() 方法。 <clinit>() 方法是由编译器自动收集类中的所有类变量(静态变量)的赋值动作和静态语句块( static{} )中的语句合并产生的。JVM会保证这个方法在多线程环境下被正确地加锁和同步,即只有一个线程能够执行这个类的 <clinit>() 方法。

 

 

Java类加载机制

 

双亲委派模型(Parents Delegation Model)

工作原理:当一个类加载器收到类加载请求时,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成。只有当父类加载器反馈自己无法完成这个加载请求(它的搜索范围中没有找到对应的类)时,子加载器才会尝试自己去加载。这种层次结构就像一个树状结构,最顶层是启动类加载器(Bootstrap ClassLoader),它主要负责加载 <JAVA_HOME>/lib 目录下的类库,如 rt.jar 等核心库;然后是扩展类加载器(Extension ClassLoader),负责加载 <JAVA_HOME>/lib/ext 目录下的类库;最后是应用程序类加载器(Application ClassLoader),负责加载用户类路径( classpath )下的类。

优势:

避免类的重复加载。因为父加载器已经加载过的类,子加载器就不需要再次加载了。

保证了Java核心库的安全性。例如,用户自定义了一个 java.lang.String 类,由于双亲委派模型,这个类不会被加载,因为启动类加载器会首先加载Java核心库中的 java.lang.String 类,这样就防止了用户恶意篡改核心类库的行为。

自定义类加载器(Custom Class Loader)

在某些情况下,我们可能需要自定义类加载器。例如,从加密的字节码文件中加载类,或者从非标准的位置(如数据库)加载类。要实现自定义类加载器,需要继承 java.lang.ClassLoader 类,并重写 findClass 方法。在 findClass 方法中,需要实现从自定义的源获取字节码数据,然后调用 defineClass 方法将字节码转换为 Class 对象。通过自定义类加载器,我们可以更加灵活地控制类的加载过程,满足特殊的应用需求。


http://www.kler.cn/a/406689.html

相关文章:

  • 论文浅尝 | MindMap:知识图谱提示激发大型语言模型中的思维图(ACL2024)
  • Python中Tushare(金融数据库)入门详解
  • 【爬虫】Firecrawl对京东热卖网信息爬取(仅供学习)
  • el-select 和el-tree二次封装
  • 深入理解TensorFlow中的形状处理函数
  • django从入门到精通(六)——auth认证及自定义用户
  • python如何使用spark操作hive
  • 基于深度学习的车牌检测系统的设计与实现(安卓、YOLOV、CRNNLPRNet)+文档
  • 如何通过ChatGPT提高自己的编程水平
  • C++设计模式行为模式———中介者模式
  • STM32通过8位并口驱动TFT-1.8寸屏(ST7735)显示器
  • TESSY单元测试工具详解与操作演示:ISO 26262合规性、自定义测试用例、详细测试报告等
  • C++游戏开发详解:从核心概念到实践
  • STM32WB55RG----FUS和stack更新
  • 计算机网络socket编程(4)_TCP socket API 详解
  • 详解Java之Spring MVC篇二
  • SQL99版外连接
  • 【SQL Server】华中农业大学空间数据库实验报告 实验九 触发器
  • VSCode快速生成vue组件模版
  • BEV:显示相机视角转换-----FastBEV/IPM与LSS
  • Unity Inspector窗口可编辑的脚本变量
  • day06(单片机高级)PCB设计
  • 解锁业务成功:大数据和 AI 如何协作以释放战略洞察
  • mac安装Pytest、Allure、brew
  • 【图像去噪】论文精读:Pre-Trained Image Processing Transformer(IPT)
  • Spark RDD Checkpoint 常用于需要高容错性或深度依赖链优化的场景,特别是在机器学习和大数据处理过程中。