小研究 - JVM 的类装载机制
本文通过对一个类装载实例的分析,阐明了 Java虚拟机的类装载的代理机制和由此定义的命名空间,指出了类装载机制在容器/组件/抽象框架结构中的作用。
目录
1 引言
2 实例
3 分析
3.1 类装载的代理机制
3.2 Java的命名空间
3.3 解决问题
4 应用
4.1 容器组件抽象框架
4.2 类装载器和容器组件抽象框架
5 结论
1 引言
Java虚拟机 JVM)的类装载就是指将包含在类文件中的字节码装载到JVM中,并使其成为JVM一部分的过程口。JVM 的类动态装载技术能够在运行时刻动态地加载或者替换系统的某些功能模块,而不影响系统其他功能模块的正常运行。类的动态加载是 JVM 的一项非常重要的技术,是许多企业 Java技术的基础,应用它可以以相对简单、灵活的形式来构建复杂的企业级应用。类的动态加载技术和具有部分类似功能的动态链接库技术相比,具有灵活、面向对象、平台独立等优点。
2 实例
下面是有关 JVM 的类动态装载技术的一个例子,包括:一个类装载器(CL. java)、一个接口(A. java)和一个实现类B. java)。代码的主要部分如下:
/CL.java
publicclassCLextendsClassLoader{protectedClassfindClass)Stringname)throwsClassNot-
FoundException{
byte[]b=loadClassData(name);
returndefineClass(name,b,0,b.length);}privatebyte[]loadClassData(Stringname)throwsClass-
NotFoundException{//获取并返回指定类的字节码…}publicstaticvoidmain(Stringargs[])throwsException{
Classb=newCL().loadClass(arg[0]);
Aa=(A)b.newInstance();//59行
a.f()}
}
//A.java
publicinterfaceA{publicvoidf();
}
//B.java
publicclassBimplementsA {publicvoidf(){
System.out.println("B.f()");}
}
其中,实现类B不在类路径(CLASSPATH)中,并且是到运行时才给出的。用 JDK 1.2以上版本的 Java编译器编译以上代码,然后运行:
>javaCLB
如果如下:
b.f()
可见,类装载器CL已经成功地将类B装载到JVM中。JVM也成功地激活了类 B中的方法。在这个过程中,Java的类装载子系统实现上完成了三个步骤:装载、链接和初始化。其中链接又可以分为校验(Verification)、准备Prepara tion)和决定Resolution)三步。除决定外,其他步骤是严格按顺序完成的。各步骤的主要工作如下:
装载:查找和导入类或接口的二进制数据。
链接:执行下面的校验、准备和决定步骤,其中决定步骤是可选的。
校验:检查导入类或接口的二进制数据的正确性。
准备:给类的静态变量分配并初始化存储空间。
决定:将符号引用转成直接引用。
初始化:激活类的静态变量的初始化Java代码和静态Java代码块。
JVM 的类装载子系统的更多介绍可参见资料。
根据Java2的扩展机制3,将CL. class文件打包成clja文件,并将cl jar文件放在Java执行环境的扩展目录通常为<JAVA-HOME> /jre/lib/ext具体可查看运行时环境变量java ext dirs值)中。
当再次执行以上程序:
> java CL B
结果如下:(第59行已在代码中标出)
Exception in thread "main" java.lang.NoClassDef-
FoundError:A
atCL.main(CL.java:59)
抛出的例外指明类A没有找到,可是类A确实在类路径(CLASSPATH)中。而在将cl jar文件放在Java执行环境的扩展目录之前,程序运行正确。那么,从类路径CLASSPATH装载类和从Java扩展目录装载类到底有什么不同呢?
3 分析
3.1 类装载的代理机制
Java2的类装载模型是一种代理(delegation,有人译成委托)模型3.当JVM要求类装载器L1装载一个类时,L1首先将这个类装载请求转发给他的父装载器。只有当父装载器没有装载并无法装载这个类时,L1才获得装载这个类的机会。这样,所有类装载器的代理关系构成了一棵树。树的根是类的根装载器(bootstrap Class Loader),在JVM中他以"null表示。除根装载器以外的类装载器有且仅有一个父装载器。在创建一个装载器时,如果没有显式地给出父装载器,那么JVM将默认系统装载器为其父装载器。Java2的基本的类装载器代理结构如图1所示。

根Bootstrap)装载器:从sun boot class path装载运行时类库的核心代码;是JVM的一部分,没有父装载器。
扩展(Extension)装载器:从 java ext dirs 扩展目录)中装载代码;父装载器为根装载器;用纯Java代码实现。
系统(System orApplication)装载器:从java class path(CLASSPATH环境变量)装载代码;父装载器为扩展装载器;用纯Java代码实现;是用户自定义类装载器的缺省父装载器。
小应用程序(Applet)装载器:从用户指定的网络上的特定目录装载小应用程序代码;父装载器为系统装载器。
一个好的类装载器应该满足以下二个性质2
1)对于相同的类名,类装载器应该返回同一个类对象。
2)如果类装载器 L1将装载类 C的请求转给类装载器L2,那么对于以下的类或接口 T,L1和 L2应该返回同一个类对象:
aT为 C的直接超类;
b)T 为 C的直接超接口;
cT为 C的成员变量的类型;
d)T为C的成员函数或构建器的参数类型;
e)T为C的成员函数的返回类型
每个已经装载到JVM中的类都隐含有装载他的类装载器的信息。类方法getC lassLoader可以得到装载这个类的类装载器。已经装载到JVM中的类不能更改他的类装载器。一个类装载器认识的类包括他的父装载器认识的类和他自己装载的类。一个类装载器认识的类是他自己装载的类的超集。
Java2中的类的装载过程是代理装载的过程。比如:W el浏览器中的JVM需要装载一个小应用程序SampleApplet JVM调用小应用程序装载器ACL来完成装载。ACL首先请求他的父装载器,即系统装载器装载SampleApplet由于SampleA pplet不在系统装载器的装载路径中,所以系统装载器没有找到这个类,也就没有装载成功。接着ACL自己装载SampleApplet ACL通过网络成功地找到了SampleApplet class文件并将他导入到了JVM中。在装载过程中,JVM发现SampleApplet是从超类java applet Applet继承的。所以JVM再次调用ACL来装载java applet Applet类。ACL又再次按上面的顺序装载Applet类,结果ACL发现他的父装载器已经装载了这个类,所以ACL就直接将这个已经装载的类返回给了JVM,完成了Applet类的装载。接下来, Applet类的超类也一样处理。最后, SampleA pplet及所有有关的类都装载到了 JVM 中。
3.2 Java的命名空间
从类装载的代理机制可以看出:在Java中,不同的类装载器定义了不同的命名空间。并且这些由类装载器定义的命名空间会有部分重叠,这保证了面向对象技术的一些重要特性,比如继承和多态的实现同。可以想象,如果没有这种父装载器首先获得装载权利的代理机制,而是所有的类装载器都各自装载,那么各个类装载器装载到JVM中的类都相互无关。JVM 本身的一些基本的类,如java lang. Object类在每个类装载器的命名空间中都得保留一个副本。并且更为严重的是,这些不同命名空间中的类被完全隔离开来了,他们之间不能进行任何形式的交互,也就不存在继承、多态等一些面象对象的关键特性。
3.3 解决问题
理解了类装载的代理机制和Java的命名空间后,再看本文前面的实例中抛出的例外,就很显然了。将cl jar文件放在Java执行环境的扩展目录以前,从命令行运行程序 CL,JVM首先调用系统类装载器装载CL类,由于系统装载器的父装载器是扩展类装载器,所以扩展类装载器先于系统类装载器获得装载CL的机会,可是扩展装载器在他的装载目录中没有找到 CL类的定义,接着,系统类装载器获得装载 CL类的机会,由于CL类在系统路径中,所以系统类装载器最终装载了类CL。类 A在类CL中被引用,其装载过程和类CL的装载过程完全一致。
从 java命名空间的划分来看,在将 cl jar文件放在 Java执行环境的扩展目录以前和以后。类CL和类A在Java的命名空间的位置如图2所示。

对类装载的代理机制和 Java的命名空间的理解能够加深对Java语言的认识。比如:有时会听到这样的表述:“一个Java类可由它的全名包名+.+类名)唯一标识”。这种表述只有在同一命名空间中是正确的,如果要在所有命名空间中都成立,须如下表述:“一个 Java类可由它的全名包名+.+类名)和它的类装载器唯一标识”。
4 应用
4.1 容器组件抽象框架
抽象框架是容器和由容器管理的组件之间的某种约定。通常,容器为组件提供了一些公共服务,如激活、生命周期、持久性、安全、事务等回。比如:一个Java兼容的WEB浏览器给小应用程序提供了激活和生命周期服务。这时,java applet包就是抽象框架,用户写的特定的小应用程序就是组件,浏览器就是容器。注意:java applet包中的Applet类是抽象的。容器和组件间的接口的定义通常是抽象的,这也是为什么称抽象框架的原因。
4.2 类装载器和容器组件抽象框架
为了实现容器组件抽象框架这种架构,关键的一点是要确保定义抽象框架的抽象类由唯一的类装载器装载,并且对所有需要引用抽象框架的抽象类的类装载器可见。这样就保证了容器中引用的抽象框架和组件中引用的抽象框架是同一个抽象框架,也满足了面向对象的继承、多态特性。通常,定义抽象框架的抽象类由实现他的组件的类装载器的父装载器来装载。如在Servlet兼容的WEB服务器中,将装载Servle抽象框架类的某个核心类装载器作为所有 Servlet类的父装载器。容器 /组件 /抽象框架和类装载器的关系如图3所示。图3可以看出,类装载器的代理机制从逻辑上保证了容器/组件抽象框架这三者关系的正确实现。
5 结论
类的动态装载机制是 JVM 的一项核心技术,也是容易被忽视而引起很多误解的地方。只有深刻地理解了类装载的代理机制以及由此引出的Java命名空间,才能更加灵活、可靠、有效地构建复杂的企业级应用。

相关文章:
小研究 - JVM 的类装载机制
本文通过对一个类装载实例的分析,阐明了 Java虚拟机的类装载的代理机制和由此定义的命名空间,指出了类装载机制在容器/组件/抽象框架结构中的作用。 目录 1 引言 2 实例 3 分析 3.1 类装载的代理机制 3.2 Java的命名空间 3.3 解决问题 4 应…...
项目---日志系统
目录 项目系统开发环境核心技术日志系统介绍为什么需要日志系统? 日志系统框架设计日志系统模块划分代码实现通用工具实现日志等级模块实现日志消息模块实现格式化模块实现落地模块实现日志器模块同步日志器异步日志器缓冲区实现异步工作器实现 回归异步日志器模块建造者模式日…...
设计模式--建造者模式(Builder Pattern)
一、什么是建造者模式 建造者模式(Builder Pattern)是一种创建型设计模式,它关注如何按照一定的步骤和规则创建复杂对象。建造者模式的主要目的是将一个复杂对象的构建过程与其表示分离,从而使同样的构建过程可以创建不同的表示。…...
若依vue打印的简单方法
像我们后端程序员做前端的话,有时候真不需要知道什么原理,直接塞就好了 我们选用基于hiprint 的vue-plugin-hiprint来打印 目的是为了实现点击某些行的数据,然后点击某个按钮直接弹出下面的打印 此链接 大佬是原创,我拿来总结梳理一下 插件进阶功能请移步: 链接 插件模板制作页…...
Rust 基础语法学习
Rust 基础语法学习 文章目录 Rust 基础语法学习hello world变量数据类型整数类型进制表示方法浮点数类型布尔类型字符类型字符串复合类型元组结构体元组结构体 切片类型字符串切片数组切片 不可变变量与可变变量常量注释函数语句与表达式 流程控制语句if else条件判断while循环…...
iOS开发Swift-函数
1.函数的定义和调用 func greet(person: String) -> String { // 函数名 传入值 传入值类型 返回值类型let greeting "Hello" personreturn greeting } print( greet(person: "Anna") ) //调用2.函数的参数与返回值 (1)无参函数 func sayHe…...
序列化协议:JSON和XML
作者:CARROT 链接:https://www.zhihu.com/question/604811576/answer/3100483698 来源:知乎 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。 json和xml都是数据传输的格式。比如我们开发过程中需要和网…...
江西萍乡能源石油化工阀门三维扫描3d测量抄数建模-CASAIM中科广电
长期以来,石油天然气、石油石化、发电和管道输送行业在环保、健康和安全保障方面一直承受着巨大的压力,他们必须确保相关规程在各项作业中得到全面贯彻。 阀门作为流体管道运输中的组成部分,其装配密封度是保证流体运输安全的重要一环&#…...
Go【gin和gorm框架】实现紧急事件登记的接口
简单来说,就是接受前端微信小程序发来的数据保存到数据库,这是我写的第二个接口,相比前一个要稍微简单一些,而且因为前端页面也是我写的,参数类型自然是无缝对接_ 前端页面大概长这个样子 先用apifox模拟发送请求测试…...
第一个VUE程序?
<!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title></head> <body><div id"app">{{message}} </div><!-- 1.导入Vue.js --> <script s…...
电阻器件的分类
电阻器的种类碳膜电阻膜式电阻器中的一种。气态碳氢化合物在高温和真空中分解,碳沉积在瓷棒或者瓷管上,形成一层结晶碳膜。改变碳膜厚度和用刻槽的方式变更碳膜的长度可以得到不同的阻值。碳膜电阻成本较低,电性能和稳定性较差,一…...
QT基础教程之二 第一个Qt小程序
QT基础教程之二 第一个Qt小程序 按钮的创建 在Qt程序中,最常用的控件之一就是按钮了,首先我们来看下如何创建一个按钮 QPushButton * btn new QPushButton; 头文件 #include <QPushButton>//设置父亲btn->setParent(this);//设置文字btn-&g…...
Edge用户数据目录查找
创建 Microsoft Edge 用户数据目录变量...
最新外卖霸王餐小程序、H5、微信公众号版外卖系统源码|霸王餐美团/饿了么系统/外卖红包cps粉丝裂变玩法源码下载
最新外卖霸王餐小程序、H5、微信公众号版外卖系统源码、霸王餐美团、饿了么系统,粉丝裂变玩源码下载,外卖cps小程序项目,外卖红包cps带好友返利佣金分销系统程序、饿了么美团联盟源码,外卖cps带分销返利后端源码,基于L…...
数据库事务四大特性
事务的4大特性(ACID): 原子性(Atomicity): 事务是数据库的逻辑工作单位,它对数据库的修改要么全部执行,要么全部不执行。 一致性(Consistemcy): 事务前后,数据库的状态都满足所有的完…...
浅谈Router和Route
router 和 route 是在前端框架中用于管理和处理路由的两个关键概念。这两者之间的关系可以通过具体的代码来解释。在本示例中,我将使用 React 和 React Router 来说明它们之间的关系。 Router(路由器):Router 是一个库或框架&…...
Linux环境安装jdk
1.安装jdk 上传jdk.tar.gz;安装包在下载内容里可以直接下载tar -zxvf jdk.tar.gz;配置环境变量:vi /etc/profile;填入以下内容;退出编辑模式,保存;然后source /etc/profile使配置生效; export JAVA_HOME/d…...
数据隐私与安全在大数据时代的挑战与应对
文章目录 数据隐私的挑战数据安全的挑战应对策略和方法1. 合规和监管2. 加密技术3. 匿名化和脱敏4. 安全意识培训5. 隐私保护技术 结论 🎈个人主页:程序员 小侯 🎐CSDN新晋作者 🎉欢迎 👍点赞✍评论⭐收藏 ✨收录专栏&…...
vue3 基础知识 (生命周期) 06
你好! 文章目录 一、生命周期二、生命周期过程三、组件的 v-model 一、生命周期 每个组件都可能从 创建、挂载、更新、卸载 等一系列的过程 在这个过程中的某一个阶段,用于可能会想要 添加一些属于自己的代码逻辑(比如组件创建完成后请求一些…...
【Eclipse】汉化简体中文教程(官方汉化包,IDE自带软件安装功能),图文详情
目录 0.环境 1.步骤 1)查看eclipse的版本 2)在官网找语言包,并复制链接 3)将链接复制到eclipse中 4)汉化完成 0.环境 windows11,64位; eclipse 2021-6版本 1.步骤 思路:在官网找…...
多云管理“拦路虎”:深入解析网络互联、身份同步与成本可视化的技术复杂度
一、引言:多云环境的技术复杂性本质 企业采用多云策略已从技术选型升维至生存刚需。当业务系统分散部署在多个云平台时,基础设施的技术债呈现指数级积累。网络连接、身份认证、成本管理这三大核心挑战相互嵌套:跨云网络构建数据…...
ElasticSearch搜索引擎之倒排索引及其底层算法
文章目录 一、搜索引擎1、什么是搜索引擎?2、搜索引擎的分类3、常用的搜索引擎4、搜索引擎的特点二、倒排索引1、简介2、为什么倒排索引不用B+树1.创建时间长,文件大。2.其次,树深,IO次数可怕。3.索引可能会失效。4.精准度差。三. 倒排索引四、算法1、Term Index的算法2、 …...
基于Java Swing的电子通讯录设计与实现:附系统托盘功能代码详解
JAVASQL电子通讯录带系统托盘 一、系统概述 本电子通讯录系统采用Java Swing开发桌面应用,结合SQLite数据库实现联系人管理功能,并集成系统托盘功能提升用户体验。系统支持联系人的增删改查、分组管理、搜索过滤等功能,同时可以最小化到系统…...
JS设计模式(4):观察者模式
JS设计模式(4):观察者模式 一、引入 在开发中,我们经常会遇到这样的场景:一个对象的状态变化需要自动通知其他对象,比如: 电商平台中,商品库存变化时需要通知所有订阅该商品的用户;新闻网站中࿰…...
代码规范和架构【立芯理论一】(2025.06.08)
1、代码规范的目标 代码简洁精炼、美观,可持续性好高效率高复用,可移植性好高内聚,低耦合没有冗余规范性,代码有规可循,可以看出自己当时的思考过程特殊排版,特殊语法,特殊指令,必须…...
怎么让Comfyui导出的图像不包含工作流信息,
为了数据安全,让Comfyui导出的图像不包含工作流信息,导出的图像就不会拖到comfyui中加载出来工作流。 ComfyUI的目录下node.py 直接移除 pnginfo(推荐) 在 save_images 方法中,删除或注释掉所有与 metadata …...
Java数组Arrays操作全攻略
Arrays类的概述 Java中的Arrays类位于java.util包中,提供了一系列静态方法用于操作数组(如排序、搜索、填充、比较等)。这些方法适用于基本类型数组和对象数组。 常用成员方法及代码示例 排序(sort) 对数组进行升序…...
CTF show 数学不及格
拿到题目先查一下壳,看一下信息 发现是一个ELF文件,64位的 用IDA Pro 64 打开这个文件 然后点击F5进行伪代码转换 可以看到有五个if判断,第一个argc ! 5这个判断并没有起太大作用,主要是下面四个if判断 根据题目…...
简约商务通用宣传年终总结12套PPT模版分享
IOS风格企业宣传PPT模版,年终工作总结PPT模版,简约精致扁平化商务通用动画PPT模版,素雅商务PPT模版 简约商务通用宣传年终总结12套PPT模版分享:商务通用年终总结类PPT模版https://pan.quark.cn/s/ece1e252d7df...
标注工具核心架构分析——主窗口的图像显示
🏗️ 标注工具核心架构分析 📋 系统概述 主要有两个核心类,采用经典的 Scene-View 架构模式: 🎯 核心类结构 1. AnnotationScene (QGraphicsScene子类) 主要负责标注场景的管理和交互 🔧 关键函数&…...
