JVM加载class文件的原理机制

c#小王子 c#小王子 2022-03-16 471 Java

1.Java中的所有类,必须被装载到jvm中才能运行,这个装载工作是由jvm中的类装载器完成的,类装载器所做的工作实质是把类文件从硬盘读取到内存中

 

2.java中的类大致分为三种:

    1.系统类

    2.扩展类

    3.由程序员自定义的类

 

3.类装载方式,有两种


    1.隐式装载, 程序在运行过程中当碰到通过new 等方式生成对象时,隐式调用类装载器加载对应的类到jvm中,

    2.显式装载, 通过class.forname()等方法,显式加载需要的类


  隐式加载与显式加载的区别?两者本质是一样?

 

4.类加载的动态性体现

    一个应用程序总是由n多个类组成,Java程序启动时,并不是一次把所有的类全部加载后再运行,它总是先把保证程序运行的基础类一次性加载到jvm中,其它类等到jvm用到的时候再加载,这样的好处是节省了内存的开销,因为java最早就是为嵌入式系统而设计的,内存宝贵,这是一种可以理解的机制,而用到时再加载这也是java动态性的一种体现

 

5.java类装载器

    Java中的类装载器实质上也是类,功能是把类载入jvm中,值得注意的是jvm的类装载器并不是一个,而是三个,层次结构如下:

      Bootstrap Loader  - 负责加载系统类

            |

          - - ExtClassLoader  - 负责加载扩展类

                    |

                   - - AppClassLoader  - 负责加载应用类

    为什么要有三个类加载器,一方面是分工,各自负责各自的区块,另一方面为了实现委托模型,下面会谈到该模型

 

6. 类加载器之间是如何协调工作的

      前面说了,java中有三个类加载器,问题就来了,碰到一个类需要加载时,它们之间是如何协调工作的,即java是如何区分一个类该由哪个类加载器来完成呢。


在这里java采用了委托模型机制,这个机制简单来讲,就是“类装载器有载入类的需求时,会先请示其Parent使用其搜索路径帮忙载入,如果Parent 找不到,那么才由自己依照自己的搜索路径搜索类”,注意喔,这句话具有递归性

下面举一个例子来说明,为了更好的理解,先弄清楚几行代码:

Public class Test{ 
    Public static void main(String[] arg){ 
      ClassLoader c  = Test.class.getClassLoader();  //获取Test类的类加载器 
        System.out.println(c);  
      ClassLoader c1 = c.getParent();  //获取c这个类加载器的父类加载器 
        System.out.println(c1); 
      ClassLoader c2 = c1.getParent();//获取c1这个类加载器的父类加载器 
        System.out.println(c2); 
  } 
} 


运行结果:

    。。。AppClassLoader。。。

    。。。ExtClassLoader。。。

    Null


注: 。。。表示省略了内容

可以看出Test是由AppClassLoader加载器加载的

AppClassLoader的Parent 加载器是 ExtClassLoader 但是ExtClassLoader的Parent为 null 是怎么回事呵,朋友们留意的话,前面有提到Bootstrap Loader是用C++语言写的,依java的观点来看,逻辑上并不存在Bootstrap Loader的类实体,所以在java程序代码里试图打印出其内容时,我们就会看到输出为null

 

类装载器ClassLoader(一个抽象类)描述一下JVM加载class文件的原理机制


类装载器就是寻找类或接口字节码文件进行解析并构造JVM内部对象表示的组件,在java中类装载器把一个类装入JVM,经过以下步骤:


1、装载:查找和导入Class文件


2、链接:其中解析步骤是可以选择的

(a)检查:检查载入的class文件数据的正确性

(b)准备:给类的静态变量分配存储空间

(c)解析:将符号引用转成直接引用


3、初始化:对静态变量,静态代码块执行初始化工作

类装载工作由ClassLoder和其子类负责。JVM在运行时会产生三个ClassLoader:根装载器,ExtClassLoader(扩展类装载器)和AppClassLoader,其中根装载器不是ClassLoader的子类,由C++编写,因此在java中看不到他,负责装载JRE的核心类库,如JRE目录下的rt.jar,charsets.jar等。ExtClassLoader是ClassLoder的子类,负责装载JRE扩展目录ext下的jar类包;AppClassLoader负责装载classpath路径下的类包,这三个类装载器存在父子层级关系,即根装载器是ExtClassLoader的父装载器,ExtClassLoader是AppClassLoader的父装载器。默认情况下使用AppClassLoader装载应用程序的类


Java装载类使用“全盘负责委托机制”。“全盘负责”是指当一个ClassLoder装载一个类时,除非显示的使用另外一个ClassLoder,该类所依赖及引用的类也由这个ClassLoder载入;“委托机制”是指先委托父类装载器寻找目标类,只有在找不到的情况下才从自己的类路径中查找并装载目标类。这一点是从安全方面考虑的,试想如果一个人写了一个恶意的基础类(如java.lang.String)并加载到JVM将会引起严重的后果,但有了全盘负责制,java.lang.String永远是由根装载器来装载,避免以上情况发生


除了JVM默认的三个ClassLoder以外,第三方可以编写自己的类装载器,以实现一些特殊的需求。类文件被装载解析后,在JVM中都有一个对应的java.lang.Class对象,提供了类结构信息的描述。数组,枚举及基本数据类型,甚至void都拥有对应的Class对象。Class类没有public的构造方法,Class对象是在装载类时由JVM通过调用类装载器中的defineClass()方法自动构造的

 

ClassLoader重要方法:

(1)public Class<?> loadClass(String name)

                   throws ClassNotFoundException

name参数指定类装载器需要装载类的名字,必须使用全限定类名。该方法有一个重载方法loadClass(String name,Boolean resolve),resolve参数告诉类装载器是否解析该类。在初始化类之前应考虑进行类解析的工作,但并不是所有类都需要解析,如果JVM只需要知道该类是否存在或找出该类的超类,那么就不需要进行解析

(2)

 

 

【注:以下内容大部分引用java深度历险】

弄明白了上面的示例,接下来直接进入类装载的委托模型实例,写两个文件,如下:

文件:Test1.java

Public class Test1{ 
    Public static void main(String[] arg){ 
        System.out.println(Test1.class.getClassLoader()); 
        Test2 t2 = new Test2(); 
        T2.print(); 
  } 
} 

 

文件: Test2.java

Public class Test2{ 
    Public void print(){ 
        System.out.println(this.getClass().getClassLoader()); 
    } 
} 

 

这两个类的作用就是打印出载入它们的类装载器是谁, 将这两个文件保存到d:\TestClassLoder目录下,编译后,我们在复制两份,分别置于 <JRE所在目录>\classes下(没有此目录,需自己建立) 与 <JRE所在目录>\lib\ext\classes下(没此目录,手工建立), 然后切换到D:\TestClassLoder目录下开始测试(查看当前用的jdk版本号,我用的jre所在目录为是C:\Program Files\Java\jdk1.6.0\jre)

 

 

测试一:

<JRE所在目录>\classes下

Test1.class

Test2.class

 

<JRE所在目录>\lib\ext\classes下

Test1.class

Test2.class

 

D:\TestClassLoder下

Test1.class

Test2.class

 

dos下输入运行命令,结果如下:

D:\TestClassLoder>java Test1

Null

Null

 

D:\TestClassLoder>

    

    从输出结果我们可以看出,当AppClassLoader要载入Test1.class时,先请其Parent,也就是ExtClassLoader来载入,而ExtclassLoader又请求其Parent,即Bootstrap Loader来载入Test1.class. 由于 <JRE所在目录>\Classes目录为Bootstrap Loader的搜索路径之一,所以Bootstrap Loader找到了Test1.class,因此将它载入,接着在Test1.class之内有载入Test2.class的需求,由于 Test1.class是由Bootstrap Loader所载入,所以Test2.class内定是由Bootstrap Loader根据其搜索路径来找,因Test2.class也位于Bootstrap Loader可以找到的路径下,所以也被载入了,最后我们看到Test1.class与Test2.class都是由Bootstrap Loader(null)载入。

 

若<JRE所在目录>\lib\ext\classes下没有这两个类文件,结果也一样都为null

 

测试二:

<JRE所在目录>\classes下

Test1.class

 

<JRE所在目录>\lib\ext\classes下

Test1.class

Test2.class

 

D:\TestClassLoder下

Test1.class

Test2.class

 

dos下输入运行命令,结果如下:

D:\TestClassLoder>java Test1

Null

Exception in thread “main” java.lang.NoClassdefFoundError:Test2 at Test1.main。。。

D:\TestClassLoder>

 

    从输出结果我们可以看出,当AppClassLoader要载入Test1.class时,先请其Parent,也就是ExtClassLoader来载入,而ExtclassLoader又请求其Parent,即Bootstrap Loader来载入Test1.class. 由于 <JRE所在目录>\Classes目录为Bootstrap Loader的搜索路径之一,所以Bootstrap Loader找到了Test1.class,因此将它载入,接着在Test1.class之内有载入Test2.class的需求,由于 Test1.class是由Bootstrap Loader所载入,所以Test2.class内定是由Bootstrap Loader根据其搜索路径来找,但是因为Bootstrap Loader根本找不到Test2.class(被我们删除了),而Bootstrap Loader又没有Parent,所以无法载入Test2.class.最后我们看到Test1.class是由Bootstrap Loader(null)载入,而Test2.class则无法载入

 

若<JRE所在目录>\lib\ext\classes下没有这两个类文件,结果也一样

 

 

测试三

<JRE所在目录>\classes下

 

Test2.class

 

<JRE所在目录>\lib\ext\classes下

Test1.class

Test2.class

 

D:\TestClassLoder下

Test1.class

Test2.class

 

dos下输入运行命令,结果如下:

D:\TestClassLoder>java Test1

。。。ExtClassLoader。。。

Null

 

D:\TestClassLoder>

 

    从输出结果我们可以看出,当AppClassLoader要载入Test1.class时,先请其Parent,也就是ExtClassLoader来载入,而ExtclassLoader又请求其Parent,即Bootstrap Loader来载入Test1.class.但是Bootstrap Loader无法在其搜索路径下找到Test1.class(被我们删掉了),所以ExtClassLoader只得自己搜索,因此 ExtClassLoader在其搜索路径 <JRE所在目录>\lib\ext\classes下找到了Test1.class,因此将它载入,接着在Test1.class之内有载入Test2.class的需求,由于Test1.class是由ExtClassLoader所载入,所以Test2.class内定是由 ExtClassLoader根据其搜索路径来找,但是因为ExtClassLoader有Parent,所以先由Bootstrap Loader帮忙寻找,Test2.class位于Bootstrap Loader可以找到的路径下,所以被Bootstrap Loader载入了.最后我们看到Test1.class是由ExtClassLoader载入,而Test2.class则是由Bootstrap Loader(null)载入

 

    了解了以上规则,请朋友们自行分析以下场景的执行结果

 

测试四:

<JRE所在目录>\classes下

 

<JRE所在目录>\lib\ext\classes下

Test1.class

Test2.class

 

D:\TestClassLoder下

Test1.class

Test2.class

 

 

测试五:

<JRE所在目录>\classes下

 

<JRE所在目录>\lib\ext\classes下

Test1.class

 

D:\TestClassLoder下

Test1.class

Test2.class

 

 

测试六:

<JRE所在目录>\classes下

 

<JRE所在目录>\lib\ext\classes下

Test2.class

 

D:\TestClassLoder下

Test1.class

Test2.class

 

 

测试七:

<JRE所在目录>\classes下

 

<JRE所在目录>\lib\ext\classes下

 

D:\TestClassLoder下

Test1.class

Test2.class

 

答案:

测试四:

。。。ExtClassLoader。。。

。。。ExtClassLoader。。。

测试五:

。。。ExtClassLoader。。。

Exception in thread "main" java.lang.NoClassDefFoundError: Test2

        at Test1.main。。。

测试六:

。。。ExtClassLoader。。。

。。。AppClassLoader。。。

测试七:

。。。AppClassLoader。。。

。。。AppClassLoader。。。

 


【下载地址】

百度网盘链接:https://pan.baidu.com/s/1YEDBIYjgWw15rtTRajMUDg

提取码:rnvb


相关文章


使用-JFreeChart来创建基于web的图表

使用-JFreeChart来创建基于web的图表

XStream使用文档

XStream使用文档

WebService发布过程及常见问题

WebService发布过程及常见问题

webpack实战入门进阶调优分享

webpack实战入门进阶调优分享

weblogic调优参数及监控指标

weblogic调优参数及监控指标

weblogic节点管理

weblogic节点管理

weblogic管理控制台概述

weblogic管理控制台概述

weblogic-部署和启动

weblogic-部署和启动

WebLogic-Server-性能及调优-调优-Java-虚拟机

Java 虚拟机(Java virtual machine,简称 JVM)是一种虚拟“执行引擎”实例,可在微处理器上执行 Java 类文件中的字节码。调整 JVM 的方式会影响 Weblogic Server 和应用程序的性能。

Velocity用户教程

Velocity是一个基于java的模板引擎(template engine)。它允许任何人仅仅简单的使用模板语言(template language)来引用由java代码定义的对象。

Velocity用户手册

Velocity 用户手册是帮助页面设计者和内容提供者认识 Velocity 和其简单而功能强大的脚本语言――Velocity 模板语言(VTL)。在手册上的许多例子,都是用 Velocity 插入动态的内容到网页上,但是所有的 VLT 例子都能应用到其他的页面和模板中。


文章热度: 166291
文章数量: 333
推荐阅读

FlashFXP绿色版网盘下载,附激活教程 1769

FlashFxp百度网盘下载链接:https://pan.baidu.com/s/1MBQ5gkZY1TCFY8A7fnZCfQ。FlashFxp是功能强大的FTP工具

Adobe Fireworks CS6 Ansifa绿色精简版网盘下载 1557

firework可以制作精美或是可以闪瞎眼的gif,这在广告领域是需要常用的,还有firework制作下logo,一些原创的图片还是很便捷的,而且fireworks用法简单,配合dw在做网站这一块往往会发挥出很强大的效果。百度网盘下载链接:https://pan.baidu.com/s/1fzIZszfy8VX6VzQBM_bdZQ

navicat for mysql中文绿色版网盘下载 1619

Navicat for Mysql是用于Mysql数据库管理的一款图形化管理软件,非常的便捷和好用,可以方便的增删改查数据库、数据表、字段、支持mysql命令,视图等等。百度网盘下载链接:https://pan.baidu.com/s/1T_tlgxzdQLtDr9TzptoWQw 提取码:y2yq

火车头采集器(旗舰版)绿色版网盘下载 1703

火车头采集器是站长常用的工具,相比于八爪鱼,简洁好用,易于配置。火车头能够轻松的抓取网页内容,并通过自带的工具对内容进行处理。站长圈想要做网站,火车头采集器是必不可少的。百度网盘链接:https://pan.baidu.com/s/1u8wUqS901HgOmucMBBOvEA

Photoshop(CS-2015-2023)绿色中文版软件下载 1819

安装文件清单(共46G)包含Window和Mac OS各个版本的安装包,从cs到cc,从绿色版到破解版,从安装文件激活工具,应有尽有,一次性打包。 Photoshop CC绿色精简版 Photoshop CS6 Mac版 Photoshop CC 2015 32位 Photoshop CC 2015 64位 Photoshop CC 2015 MAC版 Photoshop CC 2017 64位 Adobe Photoshop CC 2018 Adobe_Photoshop_CC_2018 Photoshop CC 2018 Win32 Photoshop CC 2018 Win64

知之

知之平台是全球领先的知识付费平台。提供各个领域的项目实战经验分享,提供优质的行业解决方案信息,来帮助您的工作和学习

使用指南 建议意见 用户协议 友情链接 隐私政策 Powered by NOOU ©2020 知之