你了解 Java 的类加载器吗?
Java 类加载器(ClassLoader)
Java 中的类加载器是用于加载 .class 文件到 JVM 中的组件,它的核心作用是将字节码(.class 文件)加载到内存,并且使它能够被 JVM 执行。类加载器决定了类的加载顺序和加载方式,是 Java 应用程序在运行时执行的重要部分。
1. 类加载器的基本概念
- 类加载器:Java 中的类加载器用于将
.class文件加载到 JVM 中,并将其转化为一个Class对象,之后该Class对象可以用于反射等操作。 - 每个类加载器都具备加载类的能力,但它们的加载范围和加载顺序可能不同。
2. 类加载器的类型
2.1 根类加载器(Bootstrap ClassLoader)
- 作用:是最顶层的类加载器,负责加载 Java 核心库中的类,如
java.lang.*和java.util.*。 - 实现:Bootstrap ClassLoader 是由 C++ 实现的,通常由操作系统提供,加载
jre/lib目录下的类库。 - 加载路径:由环境变量
sun.boot.class.path指定。
2.2 扩展类加载器(Extension ClassLoader)
- 作用:负责加载 JDK 中的扩展类库,主要加载
jre/lib/ext目录下的类库。 - 实现:它是由 Java 实现的,继承自
ClassLoader。 - 加载路径:由
java.ext.dirs指定。
2.3 系统类加载器(System ClassLoader)
- 作用:也叫应用类加载器,负责加载应用程序的类路径(
classpath)下的类。 - 实现:由 Java 实现,通常是用户应用程序使用的加载器,加载
classpath中的类。 - 加载路径:由环境变量
java.class.path指定。
2.4 自定义类加载器(Custom ClassLoader)
- 作用:开发人员可以自定义类加载器,继承
ClassLoader类,重写findClass()方法来加载类。 - 应用场景:
- 加载本地文件、数据库、网络等外部资源中的类。
- 用于 Web 服务器(如 Tomcat)中动态加载 JSP 文件或动态加载插件等。
3. 类加载的过程
3.1 加载(Load)
- 类加载器根据类名查找
.class文件,并读取它的字节流。 - 这个过程会依赖于类加载器的父子关系,父加载器优先加载。
3.2 链接(Linking)
链接分为三个阶段:
- 验证(Verify):检查字节码是否符合 JVM 的要求,防止非法代码进入。
- 准备(Prepare):为类的静态变量分配内存并赋默认值。
- 解析(Resolve):将常量池中的符号引用替换为直接引用。
3.3 初始化(Initialize)
- 在类的初始化阶段,JVM 执行类的静态代码块(
static块),初始化静态变量等。
4. 类加载器的双亲委派机制(Parent Delegation Model)
- 双亲委派模型是 Java 类加载器的一个重要特点,它确保了类加载的安全性和稳定性。
- 原理:
- 每个类加载器都有一个父类加载器。当一个类加载请求发生时,子加载器会先将请求传递给父加载器,父加载器先尝试加载该类。如果父加载器加载失败,子加载器再尝试加载。
- 这确保了 Java 核心类库(如
java.lang.*)始终由根类加载器加载,从而避免了重复加载和潜在的冲突。
5. 自定义类加载器的示例
通过继承 ClassLoader 类,我们可以创建自己的类加载器,下面是一个简单的自定义类加载器示例:
public class MyClassLoader extends ClassLoader {
@Override
public Class<?> findClass(String name) throws ClassNotFoundException {
byte[] classData = loadClassData(name); // 从某个源(如文件、数据库)获取字节数据
return defineClass(name, classData, 0, classData.length);
}
private byte[] loadClassData(String name) {
// 加载类的字节数据(例如,从文件或网络中读取)
return new byte[0]; // 示例返回空字节数组
}
}
public class Test {
public static void main(String[] args) throws Exception {
MyClassLoader loader = new MyClassLoader();
Class<?> clazz = loader.loadClass("com.example.MyClass");
Object obj = clazz.newInstance();
}
}
解释:
- MyClassLoader 继承自 ClassLoader,重写 findClass 方法来从某个地方加载类的字节数据。
- loadClassData 方法是我们用来加载类字节数据的地方,可以从文件、数据库等地方加载。
- defineClass 方法将字节数据转换为一个 Class 对象。
6. 类加载器的应用场景
- 动态加载插件:在应用程序运行时,动态加载新的功能模块或插件,避免重新启动应用。
- Web 服务器:如 Tomcat 会使用不同的类加载器来加载 Web 应用中的类,保证每个应用使用独立的类加载器。
- 类版本控制:在某些情况下,自定义类加载器可以用来控制不同版本的类加载,避免类冲突。
7. 总结
- 类加载器是 Java 中用于加载 .class 文件的组件,分为根类加载器、扩展类加载器、系统类加载器和自定义类加载器。
- 双亲委派机制保证了类加载的顺序和安全性。
- 自定义类加载器为我们提供了灵活性,可以从本地文件、网络等加载类,支持插件化开发。
你了解 Java 的类加载器吗?的更多相关文章
- java自定义类加载器
前言 java反射,最常用的Class.forName()方法.做毕设的时候,接收到代码字符串,通过 JavaCompiler将代码字符串生成A.class文件(存放在classpath下,也就是ec ...
- Java虚拟机类加载器及双亲委派机制
所谓的类加载器(Class Loader)就是加载Java类到Java虚拟机中的,前面<面试官,不要再问我"Java虚拟机类加载机制"了>中已经介绍了具体加载class ...
- 深入探讨java的类加载器
类加载器是 Java 语言的一个创新,也是 Java 语言流行的重要原因之一.它使得 Java 类可以被动态加载到 Java 虚拟机中并执行.类加载器从 JDK 1.0 就出现了,最初是为了满足 Ja ...
- java 中类加载器
jar 运行过程和类加载机制有关,而类加载机制又和我们自定义的类加载器有关,现在我们先来了解一下双亲委派模式. java 中类加载器分为三个: BootstrapClassLoader 负责加载 ${ ...
- java高新技术-类加载器
1.类加载器及委托机制的深入分析 > 类加载器的作用:一个java文件中的出现的类,首先要把这个类的字节码加载到内存中,这个类的信息放在硬盘的classPath下的class文件中, 把cla ...
- 深入理解Java虚拟机 - 类加载器
引子: 类加载器(classloader)是独立于虚拟机之外,可以独立实现的代码模块. OSGi使用了类加载器的这一特点实现其热插拔的特性 Java同C++等语言不通, ...
- 分析Java的类加载器与ClassLoader(二):classpath与查找类字节码的顺序,分析ExtClassLoader与AppClassLoader的源码
先回顾一下classpath classpath的作用: classpath的作用是指定查找类的路径:当使用java命令执行一个类(类中的main方法)时,会从classpath中进行查找这个类. 指 ...
- java面向对象--类加载器及Class对象
类加载器 jvm 和 类的关系 当调用 java命令运行一个java程序时,会启动一个java虚拟机进程.同一个jvm的所有线程.所有变量都处于同一个进程里,都使用该jvm进程的内存区. jvm进程终 ...
- Java的类加载器
一.类加载器的概念 类加载器(class loader)用来加载 Java 类到 Java 虚拟机中.一般来说,Java 虚拟机使用 Java 类的方式如下:Java 源程序(.java 文件)在经过 ...
- Java的类加载器种类(双亲委派)
Java类加载器采用双亲委派模型: 1.启动类加载器:这个类加载器负责放在<JAVA_HOME>\lib目录中的,或者被-Xbootclasspath参数所指定的路径中的,并且是虚拟机识别 ...
随机推荐
- Elasticsearch性能优化干货
1.集群规划优化实践 1.1 基于目标数据量规划集群 在业务初期,经常被问到的问题,要几个节点的集群,内存.CPU要多大,要不要SSD? 最主要的考虑点是:你的目标存储数据量是多大?可以针对目标数据量 ...
- Flink-cdc同步mysql到iceberg丢失数据排查
一.获取任务信息 任务id:i01f51582-d8be-4262-aefa-000000 任务名称:ods_test1234 丢失的数据时间:2024-09-16 09:28:47 二.数据同步查看 ...
- 理解ID3决策树
决策树是一个树形结构,类似下面这样: 上图除了根节点外,有三个叶子节点和一个非叶子节点. 在解决分类问题的决策树中,叶子节点就表示所有的分类,比如这里的分类就有3种:无聊时阅读的邮件.需及时处理的邮件 ...
- Failed to start: app/proxyman/inbound: failed to listen TCP on 10808
问题现象 启动 v2xxx-With-Core 失败,报错信息如下: 2023/08/03 11:38:56 [Info] infra/conf/serial: Reading config: F:\ ...
- Typora Emoji图标
转自: https://www.cnblogs.com/wangjs-jacky/p/12011208.html People :smile: :laughing: :blush: :sm ...
- Angular CLI 源码分析
准备: 安装 Node.js https://nodejs.org/: 安装 VS Code https://code.visualstudio.com/: 创建文件夹 angular-cli-sou ...
- UE5 C++ 程序进程退出
// Fill out your copyright notice in the Description page of Project Settings. #pragma once #include ...
- 【Abaqus热分析】热膨胀系数设置
来源:帮助文档
- mysql 连接失败:message from server: "Host '192.168.xx.xxx' is not allowed to connect to this MySQL server"
前言 mysql 连接失败:message from server: "Host '192.168.xx.xxx' is not allowed to connect to this MyS ...
- Golang 入门 : 整型
整型介绍 Go语言同时提供了有符号和无符号类型的整数运算.这里有 int8.int16.int32 和 int64 四种截然不同大小的有符号整形数类型,分别对应 8.16.32.64 bit大小的有符 ...