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

JVM 类加载器的工作原理

JVM 类加载器的工作原理

类加载器(ClassLoader)是一个用于加载类文件的子系统,负责将字节码文件(.class 文件)加载到 JVM 中。Java 类加载器允许 Java 应用程序在运行时动态地加载、链接和初始化类。

2. 类加载器的工作过程

JVM 类加载过程主要包括以下三个阶段:

  1. 加载(Loading)

    • 搜索并加载类文件:类加载器通过类名查找相应的 .class 文件,并将其读取到内存中。
    • 生成 Class 对象:将读取到的字节码转换成 JVM 能够识别的 Class 对象。
  2. 链接(Linking)

    • 验证(Verification):确保字节码文件的正确性和安全性,包括检查字节码格式是否正确,操作码是否正确等。
    • 准备(Preparation):为类的静态变量分配内存,并设置默认初始值。
    • 解析(Resolution):将符号引用转换为直接引用。
  3. 初始化(Initialization)

    • 执行类构造器 <clinit> 方法,这是由编译器自动生成的,用于初始化类的静态变量和静态代码块。

3. 类加载器的类型

JVM 中有几种类型的类加载器,每种类加载器有其特定的职责:

  1. 引导类加载器(Bootstrap ClassLoader)

    • 这是 JVM 自带的类加载器,用于加载 Java 核心库(即 JDK 安装目录下的 jre/lib/rt.jar 文件)。
  2. 扩展类加载器(Extension ClassLoader)

    • 加载位于 jre/lib/ext 目录中的类库或通过 java.ext.dirs 系统属性指定的类库。
  3. 应用程序类加载器(Application ClassLoader)

    • 加载应用程序的类路径(classpath)下的类文件,是用户自定义类加载的默认类加载器。
  4. 自定义类加载器(Custom ClassLoader)

    • 用户可以通过继承 ClassLoader 类并重写其方法来定义自己的类加载器。

双亲委派模型

Java 的类加载器采用双亲委派模型(Parent Delegation Model),其核心思想是:某个类加载器在加载类时,首先将类加载请求委托给父类加载器,只有在父类加载器无法完成加载时,才尝试自己加载。这一模型可以有效避免类的重复加载,确保 Java 核心类库的安全性。

双亲委派模型的工作流程

  1. 类加载请求:当应用程序需要使用一个类时,类加载器接收到该类的加载请求。
  2. 委派父加载器:当前类加载器首先将加载请求委派给它的父加载器。
  3. 递归检查:父加载器再将请求委派给它的父加载器,依次递归,直到到达引导类加载器。
  4. 加载类
    • 父加载器加载成功:如果父加载器能够找到并加载该类,则直接返回该类的 Class 对象。
    • 父加载器加载失败:如果父加载器无法加载该类,则返回给子加载器,由子加载器尝试加载。

双亲委派模型的好处

  1. 保证核心类库的安全性:通过双亲委派机制,Java 核心类库(如 java.lang.Object)由引导类加载器统一加载,避免了核心类库被篡改的风险。
  2. 避免类的重复加载:通过委派机制,可以避免同一个类被多个类加载器重复加载,从而减少内存消耗和潜在的类冲突问题。
  3. 模块化和灵活性:支持不同类加载器加载不同模块,提高了系统的模块化和灵活性。

双亲委派模型的实现

Java 类加载器通过以下几个类和方法实现双亲委派模型:

  • ClassLoader 类:Java 提供了一个抽象类 ClassLoader,所有类加载器都需要继承这个类。
  • loadClass 方法ClassLoader 类的核心方法之一,用于加载类。默认实现了双亲委派模型。
protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {synchronized (getClassLoadingLock(name)) {// 检查类是否已经加载Class<?> c = findLoadedClass(name);if (c == null) {try {// 委派父加载器加载类if (parent != null) {c = parent.loadClass(name, false);} else {c = findBootstrapClassOrNull(name);}} catch (ClassNotFoundException e) {// 父加载器未找到类}// 当前加载器尝试加载类if (c == null) {c = findClass(name);}}if (resolve) {resolveClass(c);}return c;}
}

双亲委派模型的实例

假设有一个自定义类加载器 CustomClassLoader,其父类加载器为系统类加载器。

public class CustomClassLoader extends ClassLoader {public CustomClassLoader(ClassLoader parent) {super(parent);}@Overridepublic Class<?> findClass(String name) throws ClassNotFoundException {byte[] classData = loadClassData(name);if (classData == null) {throw new ClassNotFoundException();}return defineClass(name, classData, 0, classData.length);}private byte[] loadClassData(String name) {// 自定义加载类文件字节码的逻辑return null;}
}

在加载类时,CustomClassLoader 会首先将加载请求委派给父加载器(系统类加载器),如果系统类加载器无法找到该类,才会使用 findClass 方法加载。

参考链接

  • Java 官方文档 - 类加载器

在这里插入图片描述

http://www.lryc.cn/news/372472.html

相关文章:

  • ARM Cortex-M4 CPU指令大全:作用、原理与实例
  • Mysql学习(九)——存储引擎
  • TFT屏幕波形显示
  • 服务器无法远程桌面连接不上的问题排查与解决方案
  • JAVA面试题整理——内存溢出与内存泄露的区别与联系
  • L50--- 104. 二叉树的最大深度(深搜)---Java版
  • Linux 中 “ 磁盘、进程和内存 ” 的管理
  • test_pipeline
  • 使用甲骨文云arm服务器安装宝塔时nginx无法卸载
  • C++青少年简明教程:C++的指针入门
  • Apache Doris 基础 -- 数据表设计(分层存储)
  • 使用Spring Boot设计一套BI系统
  • 2024.6.12总结
  • 1027 - 求任意三位数各个数位上数字的和
  • K8s 卷快照类
  • 从零手写实现 nginx-23-directive IF 条件判断指令
  • 08_基于GAN实现人脸图像超分辨率重建实战_超分辨基础理论
  • React.ReactElement 与 React.ReactNode
  • 深度解析服务发布策略之蓝绿发布
  • 【Mysql】 深入理解MySQL的执行计划
  • 说下你对Spring IOC 的理解
  • 前缀和算法:算法秘籍下的数据预言家
  • 基于PointNet / PointNet++深度学习模型的激光点云语义分割
  • LabVIEW调用DLL时需注意的问题
  • 时序预测 | MATLAB实现TCN-Attention自注意力机制结合时间卷积神经网络时间序列预测
  • 上位机图像处理和嵌入式模块部署(h750 mcu vs f407)
  • Linux C语言:指针和指针变量
  • Llama模型家族之Stanford NLP ReFT源代码探索 (二)Intervention Layers层
  • MATLAB神经网络---序列输入层sequenceInputLayer
  • 使用CSS、JavaScript、jQuery三种方式实现手风琴效果