Java_类加载器

server/2025/2/7 10:33:08/

小程一言

本专栏是对Java知识点的总结。在学习Java的过程中,学习的笔记,加入自己的思考,结合各种资料的整理。

文章与程序一样,一定都是不完美的,因为不完美,才拥有不断追求完美的动力

类加载器的基础

首先要明确类加载器的职责就是加载类。当JVM启动时,系统会使用类加载器将必要的类(如java.lang.*类库)加载到内存中。Java中的类加载器是递归结构,遵循“双亲委派模型”。

双亲委派模型

核心思想

当一个类加载器接到加载类的请求时,它不会立即去加载该类,而是先将请求委派给它的父类加载器。只有当父类加载器无法完成加载时,子类加载器才会尝试自己加载。

优势

它保证了Java的核心类库(如java.lang.String)始终由系统的启动类加载器来加载,避免了“重复加载”问题,并增加了类加载的安全性。

双亲委派模型的加载顺序:

  1. 启动类加载器
  2. 扩展类加载器
  3. 系统类加载器
  4. 自定义类加载器
    在这里插入图片描述

类加载器的职责

  • 启动类加载器(Bootstrap ClassLoader):负责加载JVM的核心类库,如java.lang.*包中的类。它是Java中最底层的类加载器,通常由C或C++编写,并且由JVM本身提供。

  • 扩展类加载器(Extension ClassLoader):负责加载jre/lib/ext目录下的类或JVM扩展目录中的类。它的父类是启动类加载器

  • 系统类加载器(System ClassLoader):负责加载应用程序的类路径(Classpath)中指定的类。它通常是由用户定义的类所在的目录、JAR包等。

  • 自定义类加载器(Custom ClassLoader):由开发者自定义的类加载器,可以扩展现有的类加载器,来加载类文件、网络上的类、数据库中的类等。

类加载器的工作流程

类加载器的工作流程可以简单分为以下几个步骤:

  1. 加载类加载器根据类名,查找并读取相应的字节码文件(.class文件)。
  2. 验证:JVM会验证字节码文件的有效性,确保它符合Java的语言规范。
  3. 准备:为类的静态字段分配内存,并为它们赋默认值(如null0等)。
  4. 解析:将类中的符号引用转换为直接引用,主要是指将常量池中的符号,如类名、字段名、方法名等,解析为内存地址。
  5. 初始化:执行类的静态初始化代码(如静态变量的初始化和静态块的执行)。
    在这里插入图片描述

举例:如何在Java中使用类加载器

启动类加载器、扩展类加载器与系统类加载器

java">public class ClassLoaderExample {public static void main(String[] args) {// 获取当前类加载器ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();System.out.println("系统类加载器:" + systemClassLoader);// 获取扩展类加载器ClassLoader extClassLoader = systemClassLoader.getParent();System.out.println("扩展类加载器:" + extClassLoader);// 获取启动类加载器ClassLoader bootstrapClassLoader = extClassLoader.getParent();System.out.println("启动类加载器:" + bootstrapClassLoader);}
}

输出

系统类加载器:sun.misc.Launcher$AppClassLoader@18b4aac2
扩展类加载器:sun.misc.Launcher$ExtClassLoader@5e91993f
启动类加载器:null

解释

自定义类加载器

通过继承ClassLoader类来实现自定义类加载器。假设我们有一个自定义的MyClassLoader,它从特定的目录加载类:

java">import java.io.*;
public class MyClassLoader extends ClassLoader {private String classPath;public MyClassLoader(String classPath) {this.classPath = classPath;}@Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {String path = classPath + name.replace('.', File.separatorChar) + ".class";File file = new File(path);if (file.exists()) {try (FileInputStream inputStream = new FileInputStream(file)) {byte[] bytes = new byte[(int) file.length()];inputStream.read(bytes);return defineClass(name, bytes, 0, bytes.length);} catch (IOException e) {e.printStackTrace();}}return super.findClass(name);}public static void main(String[] args) throws Exception {MyClassLoader myClassLoader = new MyClassLoader("path/to/classes/");// 使用自定义类加载器加载类Class<?> clazz = myClassLoader.loadClass("com.example.MyClass");System.out.println("加载的类: " + clazz.getName());}
}

本例中,定义了一个MyClassLoader,它从指定的路径加载类文件。当你调用loadClass()时,它会尝试使用findClass()方法加载类。如果找不到,系统类加载器会被调用。

例如,在使用自定义类加载器加载完类后,如果不再需要这些类,类加载器本身也会被垃圾回收,类将被卸载。类卸载通常发生在类加载器不再有任何引用,并且类加载器本身被回收时。
在这里插入图片描述

类加载器与类冲突

有时会遇到不同的类加载器加载同一个类的情况。如下:

java">public class TestClass {public void printMessage() {System.out.println("Hello from TestClass!");}public static void main(String[] args) throws Exception {MyClassLoader myClassLoader1 = new MyClassLoader("path/to/classes/");MyClassLoader myClassLoader2 = new MyClassLoader("path/to/classes/");Class<?> clazz1 = myClassLoader1.loadClass("TestClass");Class<?> clazz2 = myClassLoader2.loadClass("TestClass");System.out.println("clazz1 == clazz2: " + (clazz1 == clazz2));}
}

在这个例子中,我们创建了两个不同的MyClassLoader,并用它们分别加载TestClass类。由于每个MyClassLoader加载的是不同的TestClass实例,clazz1 == clazz2将会输出false,表明它们是不同的类。
在这里插入图片描述

总结

  • Java的类加载器机制确保了类的加载、验证、链接和初始化的顺序。
  • 类加载器采用双亲委派模型,从而保证了系统类库的统一加载,避免了类的重复加载。
  • 你可以通过继承ClassLoader类来自定义类加载器,控制类的加载过程。
  • 类加载器的生命周期与JVM的生命周期相匹配,因此类会在合适的时候被卸载,防止内存泄漏。

http://www.ppmy.cn/server/165654.html

相关文章

补小识JVM中的分代收集理论

JVM的分代收集理论是一套符合大多数程序运行实际情况的经验法则。它基于两个主要的分代假说&#xff1a;弱分代假说和强分代假说。弱分代假说认为&#xff0c;绝大多数对象都是朝生夕灭的&#xff1b;而强分代假说则认为&#xff0c;熬过越多次垃圾收集过程的对象就越难以消亡。…

AI大模型评测对比2—ChatGPT对比DeepSeek

近期&#xff0c;DeepSeek 凭借其超高的性价比与卓越性能&#xff0c;在国内外各大社交媒体平台上赚足了眼球。令人瞩目的是&#xff0c;它仅耗费 2048 块显卡&#xff0c; 600 亿美元的成本&#xff0c;便成功训练出了可与顶级模型比肩的 Deepseek - V3 模型。这与美国公布的一…

3 卷积神经网络CNN

1 Image Classification (Neuron Version) – 1.1 Observation 1 1.2 Observation 2 如果不同的receptive field需要相同功能的neuron&#xff0c;可以使这些neuron共享参数 1.3 Benefit of Convolutional Layer 2 Image Classification (Filter Version) 不用担心filter大小…

mysql8的并行复制介绍

mysql8的并行复制介绍 一、写在前面 大家平时在维护数据库时,肯定遇到过主从复制延迟的问题。传统的复制方式是单线程按顺序执行,这就像排队买票,每个事务只能一个一个处理,遇上高并发场景,等待时间就会变长。为了解决这个问题,MySQL 从 5.6 版本起就开始支持多线程复制…

ES6 对象扩展:对象简写,对象属性 表达式,扩展运算符 ...,Object.assign,Object.is,用法和应用场景

1. 对象属性简写 1.1 基本语法 // 传统写法 const name John; const age 25; const user {name: name,age: age };// ES6 简写语法 const user {name,age };1.2 实际应用场景 // 1. 函数返回对象 function createUser(name, age, email) {return {name,age,email}; }// …

visual studio安装

一、下载Visual Studio 访问Visual Studio官方网站。下载 Visual Studio Tools - 免费安装 Windows、Mac、Linux 在主页上找到并点击“下载 Visual Studio”按钮。 选择适合需求的版本&#xff0c;例如“Visual Studio Community”&#xff08;免费版本&#xff09;&#x…

Clion开发STM32时使用stlink下载程序与Debug调试

一、下载程序 先创建一个文件夹&#xff1a; 命名&#xff1a;stlink.cfg 写入以下代码: # choose st-link/j-link/dap-link etc. #adapter driver cmsis-dap #transport select swdsource [find interface/stlink.cfg]transport select hla_swdsource [find target/stm32f4x.…

零碎的知识点(十二):“期望” 是什么?

“期望” 是什么&#xff1f; 期望&#xff08;数学期望&#xff09;的详细解释与示例1. 期望的定义2. 期望的意义 3. 计算步骤与示例示例1&#xff1a;掷骰子的期望示例2&#xff1a;抽奖活动的期望收益示例3&#xff1a;投资决策的期望回报 4. 期望的性质5. 期望的常见误区6.…