delphi7 获取dll的类_跟我学Java内存管理----JMM精华终章(类加载器)

1 类加载器

1.1 类的加载过程

当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通过加载,连接,初始化三步来实现对这个类进行初始化。

(1)加载

就是指将class文件读入内存,并为之创建一个Class对象。

任何类被使用时系统都会建立一个Class对象。

(2)连接

验证 是否有正确的内部结构,并和其他类协调一致

准备 负责为类的静态成员分配内存,并设置默认初始化值

解析 将常量池内的符号引用(逻辑地址)替换成直接引用(物理地址)

(3)初始化 就是我们以前讲过的初始化步骤

类初始化时机

l 创建类的实例

l 访问类的静态变量,或者为静态变量赋值

l 调用类的静态方法

l 使用反射方式来强制创建某个类或接口对应的java.lang.Class对象

l 初始化某个类的子类

l 直接使用java.exe命令来运行某个主类

d4a11c85605636cd809dc5ef5da19685.png

在类加载时,静态变量和静态代码块的加载顺序(执行顺序),由编写先后决定。

在实例化时,成员变量和成员代码块的执行顺序,由编写先后决定。

1.2 什么是类加载器

类加载器就是用来加载类的东西,类加载器也是一个类:ClassLoader(把.class文件加载到JVM的方法区中,变成一个Class对象)。一些在java代码中动态生成的类,而这些类的数据就是在运行期时由类加载器去加载的,比如动态代理。

类加载器也可以被加载到内存,是通过其他类加载器完成的!Java提供了三种类加载器(比喻,它们都是片警!),分别是:

l bootstrap classloader:引导类加载器,加载rt.jar中的类(类库);

l sun.misc.Launcher$ExtClassLoader:扩展类加载器,加载lib/ext目录下的类;

l sun.misc.Launcher$AppClassLoader:应用类加载器,加载CLASSPATH下的类,即我们写的类和第三方的jar包,以及第三方提供的类。(会先加载CLASSPATH下的类,然后加载lib目录下的jar包;)

通常情况下,Java中所有类都是通过这三个类加载器加载的。

类加载器之间存在上下级关系,应用类加载器的上级是扩展类加载器,而扩展类加载器的上级是引导类加载器,引导类加载器(该加载器本身无需加载,存在JVM中)没上层,它是BOSS。

当执行 java ***.class 的时候, java.exe 会帮助我们找到 JRE ,接着找到位于 JRE 内部的 jvm.dll ,这才是真正的 Java 虚拟机器 , 最后加载动态库,激活 Java 虚拟机器。虚拟机器激活以后,会先做一些初始化的动作,比如说读取系统参数等。一旦初始化动作完成之后,就会产生第一个类加载器―― Bootstrap Loader , Bootstrap Loader 是由 C++ 所撰写而成,这个 Bootstrap Loader 所做的初始工作中,除了一些基本的初始化动作之外,最重要的就是加载 Launcher.java 之中的 ExtClassLoader ,并设定其 Parent 为 null ,代表其父加载器为 BootstrapLoader 。然后 Bootstrap Loader 再要求加载 Launcher.java 之中的 AppClassLoader ,并设定其 Parent 为之前产生的 ExtClassLoader 实体。这两个加载器都是以静态类的形式存在的。这里要请大家注意的是, Launcher$ExtClassLoader.class 与 Launcher$AppClassLoader.class 都是由 BootstrapLoader 所加载,所以 Parent 和由哪个类加载器加载没有关系。

1.3 JVM眼中的相同的类

在JVM中,不可能存在一个类被加载两次的事情!一个类如果已经被加载了,当再次试图加载这个类时,类加载器会先去查找这个类是否已经被加载过了,如果已经被加载过了,就不会再去加载了。

但是,如果一个类使用不同的类加载器去加载是可以出现多次加载的情况的!也就是说,在JVM眼中,相同的类需要有相同的class文件,以及相同的类加载器。当一个class文件,被不同的类加载器加载了,JVM会认识这是两个不同的类,这会在JVM中出现两个相同的Class对象!甚至会出现类型转换异常!

1.4类加载器的委托机制

当系统类加载器去加载一个类时,它首先会让上级去加载,即让扩展类加载器去加载类,扩展类加载器也会让它的上级引导类加载器去加载类。如果上级没有加载成功,那么再由自己去加载!

例如我们自己写的Person类,一定是存放到CLASSPATH中,那么一定是由系统类加载器来加载。当系统类加载器来加载类时,它首先把加载的任务交给扩展类加载去,如果扩展类加载器加载成功了,那么系统类加载器就不会再去加载。这就是代理模式了!

相同的道理,扩展类加载器也会把加载类的任务交给它的“上级”,即引导类加载器,引导类加载器加载成功,那么扩展类加载器也就不会再去加载了。引导类加载器是用C语言写的,是JVM的一部分,它是最上层的类加载器了,所以它就没有“上级了”。它只负责去加载“内部人”,即JDK中的类,但我们知道Person类不是我们自己写的类,所以它加载失败。

当扩展类加载器发现“上级”不能加载类,它就开始加载工作了,它加载的是libext目录下的jar文件,当然,它也会加载失败,所以最终还是由系统类加载器在CLASSPATH中去加载Person,最终由系统类加载器加载到了Person类。

代理模式保证了JDK中的类一定是由引导类加载加载的!这就不会出现多个版本的类,这也是代理模式的好处。例如自定义的String类永远不会被加载。

1.5 自定义类加载器

我们也可以通过继承ClassLoader类来完成自定义类加载器,自类加载器的目的一般是为了加载网络上的类,因为这会让class在网络中传输,为了安全,那么class一定是需要加密的,所以需要自定义的类加载器来加载(自定义的类加载器需要做解密工作)。

ClassLoader加载类都是通过loadClass()方法来完成的,loadClass()方法的工作流程如下:

l 调用findLoadedClass()方法查看该类是否已经被加载过了(在JVM的方法区中查看已加载过的类!),如果该没有加载过,那么这个方法返回null;

l 判断findLoadedClass()方法返回的是否为null,如果不是null那么直接返回,这可以避免同一个类被加载两次;

l 如果findLoadedClass()返回的是null,那么就启动代理模式(委托机制),即调用上级的loadClass()方法,获取上级的方法是getParent(),当然上级可能还有上级,这个动作就一直向上走;

l 如果getParent().loadClass()返回的不是null,这说明上级加载成功了,那么就加载结果;

l 如果上级返回的是null,这说明需要自己出手了,这时loadClass()方法会调用本类的findClass()方法来加载类;

l 这说明我们只需要重写ClassLoader的findClass()方法,这就可以了!如果重写了loadClass()方法覆盖了代理模式!

OK,通过上面的分析,我们知道要自定义一个类加载器,只需要继承ClassLoader类,然后重写它的findClass()方法即可。那么在findClass()方法中我们要完成哪些工作呢?

l 找到class文件,把它加载到一个byte[]中;

l 调用defineClass()方法,把byte[]传递给这个方法即可。

1.6Tomcat的类加载器

Tomcat提供了两种类加载器!

* 服务器类加载器:它负责加载这个下面的类!(${CATALINA_HOME}lib)

* 应用类加载器:它负责加载这两个路径下的类!(${CONTEXT_HOME}WEB-INFlib、${CONTEXT_HOME}WEB-INFclasses)。

注意:Tomcat会为每个应用(假设该Tomcat下部署了多个应用)提供一个应用类加载器,负责加载每个该应用下WEB-INFlib和WEB-INFclasses目录下的类。

对于运行在 Java EE容器中的 Web 应用来说,类加载器的实现方式与一般的 Java 应用有所不同。不同的 Web 容器的实现方式也会有所不同。以 Apache Tomcat 来说,每个 Web 应用都有一个对应的类加载器实例。该类加载器也使用代理模式,所不同的是它是首先尝试去加载某个类,如果找不到再代理给父类加载器去加载。这与一般类加载器的顺序是相反的。这是 Java Servlet 规范中的推荐做法,其目的是使得 Web 应用自己的类的优先级高于 Web 容器提供的类。

这种代理模式的一个例外是:Java 核心库的类是不在查找范围之内的。这也是为了保证 Java 核心库的类型安全。(tomcat的webApp类加载器,在加载用户自定义类时,先尝试让App类加载器判断下是不是jdk的核心类,如果是,就由App类加载器加载;否则就自己来加载;如果自己还加载不了,就交给服务器类加载器去加载。)

8bf49e27123b2542efe12cbcce958f6e.png
23b7ddf268ae46929422b5324acf3bd6.png

Tomcat提供的类加载器有这样一个好处,就是对每个应用来说,可以使该应用下的类优先被加载!(优先于(${CATALINA_HOME}lib目录下的类)。

270437156648ff066e181051caa323dd.png

相同名称的类可以并存在 Java 虚拟机中,只需要用不同的类加载器来加载它们即可。不同类加载器加载的类之间是不兼容的,这就相当于在 Java 虚拟机内部创建了一个个相互隔离的 Java 类空间。这种技术在许多框架中都被用到,例如OSGI、Web 容器(Tomcat)。

1.7加载资源文件

我们在开发中,要读取配置文件时,为什么不使用File的形式读取?因为应用有可能被打包成jar文件,单纯地用File去读取jar包的文件是不能的,因为File的方式并不会深入到jar包内部的路径(即不会解压jar文件),那么就无法读取到jar内部的配置文件了。所以,如果jar包中的类源代码用File f=new File(相对路径)的形式,是不可能定位到文件资源的,会报FileNotFoundException的异常信息。

而如果用ClassLoader来加载资源文件,就能解决这个问题。ClassLoader会使用文件资源定位符的格式 (jar中资源有其专门的URL形式: jar:!/{entry} )。

e82561339357376ee9ac88c551910007.png

1.8怎样获取ClassLoader呢

Thread.currentThread().getContextClassLoader()和ClassLoader.getSystemClassLoader()如何选择?

e8e5c1861f8aba5c58f0cd8495c30051.png

分别在IDEA中运行、以jar的方式运行上面Controller的代码查看结果。

在IDEA开发工具中运行时直接调用ClassLoader.getSystemClassLoader()调用的是AppClassLoader,在IDEA环境时可以加载到target目录下的所有文件。而如果在springBoot项目的 jar方式运行时仍然还是用AppClassLoader去加载项目内的静态资源的话就会找不到的,因为springBoot加载静态资源文件是用的内置tomcat的classloader去加载的。

总结:

为了避免在项目中加载不到本项目中静态资源文件的BUG发生,调用静态资源的classLoader最好用Thread.currentThread().getContextClassLoader()方法来获取,因为一般同一个项目中java代码和其静态资源文件都是同一个classLoader来加载的,以此确保通过此classLoader也能加载到本项目中的资源文件。

遗留知识

1.SPI机制

2.字节码技术

3.SpringBoot的spring.factories原理

4.Dubbo的SPI扩展原理

预知后事如何,且听下回分解。

JMM的内容很多很深,已经尽量浓缩了。打字不容易,记得关注——收藏——转发哦。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/507749.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

huffman树_笃学不倦|c语言构造哈夫曼树哈夫曼编码

艾薇巴蒂!许久不见甚是想念,想必这”涨姿势”的时刻大家已经期待许久了!今天我们要共同学习的是c语言构造哈夫曼树-哈夫曼编码构造哈夫曼树首先,我们需要了解哈夫曼树是什么:相关知识点路径: 路径是指从一个…

推荐系统相关科技论文写作建议

如何写标题 1、用一句话概括你所做的工作; 2、字数忌长(尽可能不要超过20单词,40-60 字符比较合适); 3、考虑搜索引擎的影响,包含关键词。 4、例子 例子1:Enhancing slope one recommendation…

睡眠音频分割及识别问题(一)

问题描述 通过手机App的录音功能,获得用户一整夜的睡眠音频,对睡眠音频进行分割,并对睡眠阶段进行判定。 (1)假设条件一:用户在相对安静的环境下进行睡眠,背景音可能会出现风声、雨声、汽车噪音…

睡眠音频分割及识别问题(四)--YAMNet简介

简介 YAMNet模型是在 AudioSet 数据集(一个大型音频、视频数据集)上训练的音频事件分类器。 模型输入 该模型接收包含任意长度波形的float32一维张量或 NumPy数组,且满足范围[-1.0, 1.0]内的单声道16kHz样本。在内部,该算法将波…

对应oracle生成java对象,Java学习笔记(十三)——通过Netbeans开发环境生成oracle数据库中表的对应hibernate映射文件...

【前面的话】身体慢慢已经快好了,感觉真好,哈哈。这篇文章要通过Hibernate对数据库进行操作,而Netbeans可以直接通过数据库逆向生成对应的映射文件。基础文章,选择性阅读。【步骤】1、 在netbeans中选择服务,点击数据库…

qt如何实现backspace的功能_如何实现知识星球列表拖拽功能

本 篇讲解一下如何实现在知识星球中星球列表的拖拽,各位可能都知道,我创建了一个星球【Hi Android】,所以最近使用这个软件比较多,之前也写过关于这个软件如何生成分享卡的文章,就是那篇文章有小伙伴建议我写一个星球的…

睡眠音频分割及识别问题(五)--YAMNet进一步分析

简介 YAMNet 是一个经过预训练的深度网络,可基于 AudioSet-YouTube 语料库 预测 521 种音频事件类别,并采用Mobilenet_v1深度可分离卷积架构。 输入 模型训练所使用的音频特征计算方式如下: 所有音频均重采样为 16 kHz 单声道。 通过长度…

vb.net调用oracle存储过程,vbnet2008连接oracle增删改查学习笔记(经典crud_含存储过程).doc...

vbnet2008连接oracle增删改查学习笔记(经典crud_含存储过程).doc 我的VBNETORACLE增删改查学习笔记(本源码在VBNET2008下测试通过)学习VBNET有一段时间了,之前一直学习VB60。过度到NET后发现与之前所学习的有了相当大的变化。于是将编程经常用到的增删改查代码提炼出…

睡眠音频分割及识别问题(六)--输入输出及方案讨论

简介 2021年7月13日,我和我的三个研究生一起拜访了玉米树,和王总等一起针对睡眠音频分割及识别问题进行了深入的讨论,达成了如下共识。 输入 由于保存整个晚上的睡眠音频所需要的存储空间过大,目前拟采用每隔30分钟&#xff08…

睡眠音频分割及识别问题(七)--接口输入输出讨论

简介 关于接口输入输出的讨论。 输入 1、音频文件路径,类型为:字符串; 2、预测多标签类型的排名,前n名,类型为:整型。 处理 1、将输入的音频分割为多个以1s为时间单位的音频片段; 2、利用P…

dhcp工具_网络分析之DHCP服务闯入QinQ二层隧道引发故障

一、Wireshark显示过滤器和QinQ二层隧道简述1.本段主要简述什么是Wireshark显示过滤器。显示过滤器是在现有的数据包中通过过滤条件,筛选想要查看的对象,不会丢失数据包,只是为了增强用户阅读而将一部分数据包隐藏起来。在“应用显…

php redis存储位置,redis数据保存在哪里

redis的数据是存在内存里吗?首先要明白redis是一个数据库,redis是一个内存数据库,所有数据基本上都存在于内存当中,会定时以追加或者快照的方式刷新到硬盘中。 (推荐学习:Redis视频教程)由于redis是一个内存数据库&…

python 多条件 选择 算法_浅析Python中的多条件排序实现

多条件排序及itemgetter的应用曾经客户端的同事用as写一大堆代码来排序,在得知Python排序往往只需要一行,惊讶无比,遂对python产生浓厚的兴趣。 之前在做足球的积分榜的时候需要用到多条件排序,如果积分相同,则按净胜球…

推荐系统: 数据、问题与算法

网络的迅速发展带来了信息量的大幅增长,使得用户在面对大量信息时无法从中获得对自己真正有用的那部分信息,对信息的使用效率反而降低了,导致信息超载(information overload)问题。 解决信息超载问题一个非常有潜力的办…

睡眠音频分割及识别问题(八)--数据采集

问题 在采用PANN或者YAMNet框架进行学习的时候,没有梦话、磨牙等睡眠音频数据,在一些公开数据集上也没有找到(如果有哪位读者知道,麻烦给我在评论区留言,万分感谢)。 公开数据集包括: &#xf…

旋转矩阵公式生成器_坐标变换(8)—复特征值与旋转

1.共轭复特征值设是的实矩阵,假设是的特征值,为对应的特征向量,则同样是的特征值,而是对应的特征向量,所以,当是的实矩阵,它的复特征值以共轭复数对出现。2. rotation-scaling matrix假如,为实数…

睡眠音频分割及识别问题(九)--Android下的YAMNet

部署PANNs模型面临的问题 加载模型出错 在使用PANNs模型时,在PC端可以较好的运行,可是在Android端运行的时候,编译过程提示缺少libpytorch_jni.so文件,导致无法加载模型,无法预测。(如果有读者可以解决这个…

linux 打开上一级目录,linux开机启动过程、PATH、过滤一级目录、cd的参数、ls -lrt、命令切割日志...

第二波命令正向我方来袭 :开机启动过程、PATH、过滤一级目录、cd的参数、ls -lrt、命令切割日志1.1 linux开机启动过程1.1.1 开机自检(BIOS)-- MBR引导-- GRUB菜单--加载内核(kernel)--运行INIT进程--读取/etc/inittab配置文件--执行/etc/rc.sysinit脚本(初始化脚本…

睡眠音频分割及识别问题(十)--Java读取wav文件

简介 Waveform Audio File Format(WAVE,又或者是因为扩展名而被大众所知的 wav),是微软与 IBM公司所开发在个人电脑存储音频流的编码格式。 由于项目需要从 wav 文件中读取音频数据,现有许多框架的 API 文档参差不齐&…

python 开发板 i2s_[Craftor原创] I2S总线接口设计(Verilog)

本文有Craftor原创,转载请保留出处。 I2S是数字音频的接口,这里不用多说,请读者自己查阅相关资料。 本文中要设计的是FPGA与数字音频芯片的I2S接口时序。简单点说,就是通过FPGA向音频芯片写数据,通过的是I2S总线&#…