小研究 - 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.步骤 思路:在官网找…...

使用分级同态加密防御梯度泄漏
抽象 联邦学习 (FL) 支持跨分布式客户端进行协作模型训练,而无需共享原始数据,这使其成为在互联和自动驾驶汽车 (CAV) 等领域保护隐私的机器学习的一种很有前途的方法。然而,最近的研究表明&…...

【第二十一章 SDIO接口(SDIO)】
第二十一章 SDIO接口 目录 第二十一章 SDIO接口(SDIO) 1 SDIO 主要功能 2 SDIO 总线拓扑 3 SDIO 功能描述 3.1 SDIO 适配器 3.2 SDIOAHB 接口 4 卡功能描述 4.1 卡识别模式 4.2 卡复位 4.3 操作电压范围确认 4.4 卡识别过程 4.5 写数据块 4.6 读数据块 4.7 数据流…...

ESP32 I2S音频总线学习笔记(四): INMP441采集音频并实时播放
简介 前面两期文章我们介绍了I2S的读取和写入,一个是通过INMP441麦克风模块采集音频,一个是通过PCM5102A模块播放音频,那如果我们将两者结合起来,将麦克风采集到的音频通过PCM5102A播放,是不是就可以做一个扩音器了呢…...
2023赣州旅游投资集团
单选题 1.“不登高山,不知天之高也;不临深溪,不知地之厚也。”这句话说明_____。 A、人的意识具有创造性 B、人的认识是独立于实践之外的 C、实践在认识过程中具有决定作用 D、人的一切知识都是从直接经验中获得的 参考答案: C 本题解…...

Mysql中select查询语句的执行过程
目录 1、介绍 1.1、组件介绍 1.2、Sql执行顺序 2、执行流程 2.1. 连接与认证 2.2. 查询缓存 2.3. 语法解析(Parser) 2.4、执行sql 1. 预处理(Preprocessor) 2. 查询优化器(Optimizer) 3. 执行器…...

并发编程 - go版
1.并发编程基础概念 进程和线程 A. 进程是程序在操作系统中的一次执行过程,系统进行资源分配和调度的一个独立单位。B. 线程是进程的一个执行实体,是CPU调度和分派的基本单位,它是比进程更小的能独立运行的基本单位。C.一个进程可以创建和撤销多个线程;同一个进程中…...

C# 表达式和运算符(求值顺序)
求值顺序 表达式可以由许多嵌套的子表达式构成。子表达式的求值顺序可以使表达式的最终值发生 变化。 例如,已知表达式3*52,依照子表达式的求值顺序,有两种可能的结果,如图9-3所示。 如果乘法先执行,结果是17。如果5…...
Caliper 配置文件解析:fisco-bcos.json
config.yaml 文件 config.yaml 是 Caliper 的主配置文件,通常包含以下内容: test:name: fisco-bcos-test # 测试名称description: Performance test of FISCO-BCOS # 测试描述workers:type: local # 工作进程类型number: 5 # 工作进程数量monitor:type: - docker- pro…...
深入浅出Diffusion模型:从原理到实践的全方位教程
I. 引言:生成式AI的黎明 – Diffusion模型是什么? 近年来,生成式人工智能(Generative AI)领域取得了爆炸性的进展,模型能够根据简单的文本提示创作出逼真的图像、连贯的文本,乃至更多令人惊叹的…...
Java并发编程实战 Day 11:并发设计模式
【Java并发编程实战 Day 11】并发设计模式 开篇 这是"Java并发编程实战"系列的第11天,今天我们聚焦于并发设计模式。并发设计模式是解决多线程环境下常见问题的经典解决方案,它们不仅提供了优雅的设计思路,还能显著提升系统的性能…...