当前位置: 首页 > news >正文

小研究 - 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 的类装载机制

本文通过对一个类装载实例的分析&#xff0c;阐明了 Java虚拟机的类装载的代理机制和由此定义的命名空间&#xff0c;指出了类装载机制在容器/组件/抽象框架结构中的作用。 目录 1 引言 2 实例 3 分析 3.1 类装载的代理机制 3.2 Java的命名空间 3.3 解决问题 4 应…...

项目---日志系统

目录 项目系统开发环境核心技术日志系统介绍为什么需要日志系统? 日志系统框架设计日志系统模块划分代码实现通用工具实现日志等级模块实现日志消息模块实现格式化模块实现落地模块实现日志器模块同步日志器异步日志器缓冲区实现异步工作器实现 回归异步日志器模块建造者模式日…...

设计模式--建造者模式(Builder Pattern)

一、什么是建造者模式 建造者模式&#xff08;Builder Pattern&#xff09;是一种创建型设计模式&#xff0c;它关注如何按照一定的步骤和规则创建复杂对象。建造者模式的主要目的是将一个复杂对象的构建过程与其表示分离&#xff0c;从而使同样的构建过程可以创建不同的表示。…...

若依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

作者&#xff1a;CARROT 链接&#xff1a;https://www.zhihu.com/question/604811576/answer/3100483698 来源&#xff1a;知乎 著作权归作者所有。商业转载请联系作者获得授权&#xff0c;非商业转载请注明出处。 json和xml都是数据传输的格式。比如我们开发过程中需要和网…...

江西萍乡能源石油化工阀门三维扫描3d测量抄数建模-CASAIM中科广电

长期以来&#xff0c;石油天然气、石油石化、发电和管道输送行业在环保、健康和安全保障方面一直承受着巨大的压力&#xff0c;他们必须确保相关规程在各项作业中得到全面贯彻。 阀门作为流体管道运输中的组成部分&#xff0c;其装配密封度是保证流体运输安全的重要一环&#…...

Go【gin和gorm框架】实现紧急事件登记的接口

简单来说&#xff0c;就是接受前端微信小程序发来的数据保存到数据库&#xff0c;这是我写的第二个接口&#xff0c;相比前一个要稍微简单一些&#xff0c;而且因为前端页面也是我写的&#xff0c;参数类型自然是无缝对接_ 前端页面大概长这个样子 先用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…...

电阻器件的分类

电阻器的种类碳膜电阻膜式电阻器中的一种。气态碳氢化合物在高温和真空中分解&#xff0c;碳沉积在瓷棒或者瓷管上&#xff0c;形成一层结晶碳膜。改变碳膜厚度和用刻槽的方式变更碳膜的长度可以得到不同的阻值。碳膜电阻成本较低&#xff0c;电性能和稳定性较差&#xff0c;一…...

QT基础教程之二 第一个Qt小程序

QT基础教程之二 第一个Qt小程序 按钮的创建 在Qt程序中&#xff0c;最常用的控件之一就是按钮了&#xff0c;首先我们来看下如何创建一个按钮 QPushButton * btn new QPushButton; 头文件 #include <QPushButton>//设置父亲btn->setParent(this);//设置文字btn-&g…...

Edge用户数据目录查找

创建 Microsoft Edge 用户数据目录变量...

最新外卖霸王餐小程序、H5、微信公众号版外卖系统源码|霸王餐美团/饿了么系统/外卖红包cps粉丝裂变玩法源码下载

最新外卖霸王餐小程序、H5、微信公众号版外卖系统源码、霸王餐美团、饿了么系统&#xff0c;粉丝裂变玩源码下载&#xff0c;外卖cps小程序项目&#xff0c;外卖红包cps带好友返利佣金分销系统程序、饿了么美团联盟源码&#xff0c;外卖cps带分销返利后端源码&#xff0c;基于L…...

数据库事务四大特性

事务的4大特性&#xff08;ACID&#xff09;&#xff1a; 原子性(Atomicity)&#xff1a; 事务是数据库的逻辑工作单位&#xff0c;它对数据库的修改要么全部执行&#xff0c;要么全部不执行。 一致性(Consistemcy)&#xff1a; 事务前后&#xff0c;数据库的状态都满足所有的完…...

浅谈Router和Route

router 和 route 是在前端框架中用于管理和处理路由的两个关键概念。这两者之间的关系可以通过具体的代码来解释。在本示例中&#xff0c;我将使用 React 和 React Router 来说明它们之间的关系。 Router&#xff08;路由器&#xff09;&#xff1a;Router 是一个库或框架&…...

Linux环境安装jdk

1.安装jdk 上传jdk.tar.gz;安装包在下载内容里可以直接下载tar -zxvf jdk.tar.gz;配置环境变量&#xff1a;vi /etc/profile&#xff1b;填入以下内容&#xff1b;退出编辑模式&#xff0c;保存&#xff1b;然后source /etc/profile使配置生效&#xff1b; export JAVA_HOME/d…...

数据隐私与安全在大数据时代的挑战与应对

文章目录 数据隐私的挑战数据安全的挑战应对策略和方法1. 合规和监管2. 加密技术3. 匿名化和脱敏4. 安全意识培训5. 隐私保护技术 结论 &#x1f388;个人主页&#xff1a;程序员 小侯 &#x1f390;CSDN新晋作者 &#x1f389;欢迎 &#x1f44d;点赞✍评论⭐收藏 ✨收录专栏&…...

vue3 基础知识 (生命周期) 06

你好&#xff01; 文章目录 一、生命周期二、生命周期过程三、组件的 v-model 一、生命周期 每个组件都可能从 创建、挂载、更新、卸载 等一系列的过程 在这个过程中的某一个阶段&#xff0c;用于可能会想要 添加一些属于自己的代码逻辑&#xff08;比如组件创建完成后请求一些…...

【Eclipse】汉化简体中文教程(官方汉化包,IDE自带软件安装功能),图文详情

目录 0.环境 1.步骤 1&#xff09;查看eclipse的版本 2&#xff09;在官网找语言包&#xff0c;并复制链接 3&#xff09;将链接复制到eclipse中 4&#xff09;汉化完成 0.环境 windows11&#xff0c;64位&#xff1b; eclipse 2021-6版本 1.步骤 思路&#xff1a;在官网找…...

可靠性+灵活性:电力载波技术在楼宇自控中的核心价值

可靠性灵活性&#xff1a;电力载波技术在楼宇自控中的核心价值 在智能楼宇的自动化控制中&#xff0c;电力载波技术&#xff08;PLC&#xff09;凭借其独特的优势&#xff0c;正成为构建高效、稳定、灵活系统的核心解决方案。它利用现有电力线路传输数据&#xff0c;无需额外布…...

学校招生小程序源码介绍

基于ThinkPHPFastAdminUniApp开发的学校招生小程序源码&#xff0c;专为学校招生场景量身打造&#xff0c;功能实用且操作便捷。 从技术架构来看&#xff0c;ThinkPHP提供稳定可靠的后台服务&#xff0c;FastAdmin加速开发流程&#xff0c;UniApp则保障小程序在多端有良好的兼…...

Qwen3-Embedding-0.6B深度解析:多语言语义检索的轻量级利器

第一章 引言&#xff1a;语义表示的新时代挑战与Qwen3的破局之路 1.1 文本嵌入的核心价值与技术演进 在人工智能领域&#xff0c;文本嵌入技术如同连接自然语言与机器理解的“神经突触”——它将人类语言转化为计算机可计算的语义向量&#xff0c;支撑着搜索引擎、推荐系统、…...

视频字幕质量评估的大规模细粒度基准

大家读完觉得有帮助记得关注和点赞&#xff01;&#xff01;&#xff01; 摘要 视频字幕在文本到视频生成任务中起着至关重要的作用&#xff0c;因为它们的质量直接影响所生成视频的语义连贯性和视觉保真度。尽管大型视觉-语言模型&#xff08;VLMs&#xff09;在字幕生成方面…...

论文解读:交大港大上海AI Lab开源论文 | 宇树机器人多姿态起立控制强化学习框架(一)

宇树机器人多姿态起立控制强化学习框架论文解析 论文解读&#xff1a;交大&港大&上海AI Lab开源论文 | 宇树机器人多姿态起立控制强化学习框架&#xff08;一&#xff09; 论文解读&#xff1a;交大&港大&上海AI Lab开源论文 | 宇树机器人多姿态起立控制强化…...

高防服务器价格高原因分析

高防服务器的价格较高&#xff0c;主要是由于其特殊的防御机制、硬件配置、运营维护等多方面的综合成本。以下从技术、资源和服务三个维度详细解析高防服务器昂贵的原因&#xff1a; 一、硬件与技术投入 大带宽需求 DDoS攻击通过占用大量带宽资源瘫痪目标服务器&#xff0c;因此…...

前端调试HTTP状态码

1xx&#xff08;信息类状态码&#xff09; 这类状态码表示临时响应&#xff0c;需要客户端继续处理请求。 100 Continue 服务器已收到请求的初始部分&#xff0c;客户端应继续发送剩余部分。 2xx&#xff08;成功类状态码&#xff09; 表示请求已成功被服务器接收、理解并处…...

数据结构:泰勒展开式:霍纳法则(Horner‘s Rule)

目录 &#x1f50d; 若用递归计算每一项&#xff0c;会发生什么&#xff1f; Horners Rule&#xff08;霍纳法则&#xff09; 第一步&#xff1a;我们从最原始的泰勒公式出发 第二步&#xff1a;从形式上重新观察展开式 &#x1f31f; 第三步&#xff1a;引出霍纳法则&…...

C++11 constexpr和字面类型:从入门到精通

文章目录 引言一、constexpr的基本概念与使用1.1 constexpr的定义与作用1.2 constexpr变量1.3 constexpr函数1.4 constexpr在类构造函数中的应用1.5 constexpr的优势 二、字面类型的基本概念与使用2.1 字面类型的定义与作用2.2 字面类型的应用场景2.2.1 常量定义2.2.2 模板参数…...

二叉树-144.二叉树的前序遍历-力扣(LeetCode)

一、题目解析 对于递归方法的前序遍历十分简单&#xff0c;但对于一位合格的程序猿而言&#xff0c;需要掌握将递归转化为非递归的能力&#xff0c;毕竟递归调用的时候会调用大量的栈帧&#xff0c;存在栈溢出风险。 二、算法原理 递归调用本质是系统建立栈帧&#xff0c;而非…...