0

0

Java 中同一个类为何能被不同 ClassLoader 重复加载?

聖光之護

聖光之護

发布时间:2026-01-17 23:07:13

|

253人浏览过

|

来源于php中文网

原创

Java 中同一个类为何能被不同 ClassLoader 重复加载?

java 类加载器遵循双亲委派模型,但当使用独立的 urlclassloader 实例时,即使指向相同 jar 文件,也会生成彼此隔离的类实例——因为类的唯一性由“全限定名 + 加载它的 classloader”共同决定。

在 Java 中,一个类是否“相同”,不仅取决于其全限定名(如 example.ToBeLoaded),更关键的是加载它的 ClassLoader 实例。JVM 在运行时将 Class 对象视为“由特定 ClassLoader 定义的类型”,即使两个 Class 对象字节码完全一致、名称完全相同,只要它们由不同的 ClassLoader 加载,JVM 就会视其为两个完全无关、互不兼容的类型。

这正是你示例中输出 false false 的根本原因:

System.out.println(child2 == child1);      // false —— 不是同一个 Class 对象引用
System.out.println(child2.equals(child1));  // false —— Class.equals() 内部也基于 classloader 判等

虽然 URLClassLoader 默认以 AppClassLoader(即系统类加载器)为父加载器,但双亲委派仅影响“加载行为”的发起顺序,而非强制复用已加载的类。关键在于:cl1 和 cl2 是两个独立的 URLClassLoader 实例,它们各自维护私有的已定义类缓存(definedClasses)。当调用 cl1.loadClass("example.ToBeLoaded") 时:

  • cl1 先委托给父加载器(AppClassLoader)尝试加载;
  • 若父加载器未找到该类(例如 ToBeLoaded 不在 classpath 中),则 cl1 自行从指定 JAR 中读取字节码并调用 defineClass() 创建新 Class 对象;
  • 同理,cl2 执行相同流程,得到另一个独立的 Class 实例。

⚠️ 注意:双亲委派 ≠ 类共享。只有当父加载器成功返回一个 Class 实例(即该类已被父加载器加载过),子加载器才不会重复定义——而你的场景中,AppClassLoader 并未加载 ToBeLoaded,因此委派失败,cl1 和 cl2 各自完成定义,结果是两个隔离的类。

立即学习Java免费学习笔记(深入)”;

京点点
京点点

京东AIGC内容生成平台

下载

这种机制并非缺陷,而是关键设计特性,广泛应用于:

  • Java EE / Jakarta EE 容器(如 Tomcat、WildFly):每个 Web 应用拥有独立 WebAppClassLoader,支持不同应用依赖同一类名但不同版本的库(如 commons-lang3:3.9 vs 3.12),避免冲突;
  • OSGi 框架:实现细粒度模块隔离与热部署;
  • 插件化系统(如 IDE 插件、Jenkins 插件):保障插件间类空间独立,防止污染主程序。

✅ 正确判断“两个 Class 是否等价”的方式是:

boolean sameClass = child1 == child2; // 必须是同一对象引用(最严格)
// 或更实用的运行时检查:
boolean sameType = child1.getClassLoader() == child2.getClassLoader()
                && child1.getName().equals(child2.getName());

❌ 错误做法:仅比较类名、包名,或试图通过 instanceof 跨类加载器判别(将抛 ClassCastException)。

总结:Java 的类唯一性模型是 (className, ClassLoader) 二元组。理解这一点,是掌握模块化、容器化、热部署等高级特性的基石。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

835

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

739

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

735

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.6万人学习

C# 教程
C# 教程

共94课时 | 6.9万人学习

Java 教程
Java 教程

共578课时 | 47.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号