深入理解JVM之类加载器

1.类加载器的作用

  类加载器(class loader)是 Java中的一个很重要的概念。类加载阶段中“把一个类的全限定名来获取描述该类的二进制字节流”这个动作是属于java虚拟机外部的,而实现这个动作的代码就是“类加载器”。
  对于任意一个类,都必须由类加载器和类本身一起在java虚拟机中的唯一性,意思就是说:两个类是否“相等”,要由同一个类加载器加载的前提下才有意义,不然哪怕两个类都是同一个class文件,被同一个java虚拟机加载都不行。

2.双亲委派模型

  自JKD1.2以来,java一直保持着三层类加载器、双亲委派的类加载架构。

2.1启动类加载器(Bootstrap ClassLoader)

  这个类加载器负责加载放在<JAVA_HOME>\lib目录,或者被-Xbootclasspath参数所指定的路径中存访的,而且是JVM能够识别的类库加载到虚拟机的内存中。
  启动类加载器无法被Java程序直接引用,用户在编写自定义类加载器时,如果需要把加载请求委派给引导类加载器去处理,那直接使用null代替即可。

2.2扩展类加载器(Extension Class Loader)

  这个类加载器是在类sun.misc.Launcher$ExtClassLoader中以Java代码的形式实现的。它负责加载<JAVA_HOME>\lib\ext目录中,或者被java.ext.dirs系统变量所指定的路径中所有的类库。

  由于扩展类加载器是由Java代码实现的,开发者可以直接在程序中使用扩展类加载器来加载Class文件。

2.3应用程序类加载器(Application Class Loader)

  这个类加载器由sun.misc.Launcher$AppClassLoader来实现。由于应用程序类加载器是ClassLoader类中的getSystemClassLoader()方法的返回值,所以有些场合中也称它为“系统类加载器”。它负责加载用户类路径(ClassPath)上所有的类库,开发者同样可以直接在代码中使用这个类加载器。

2.4双亲委派模型

在这里插入图片描述
  图中各自类加载器之间的层次关系被称为类加载器的“双亲委派模型”。
  双亲委派模型的工作过程是:如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的类加载器都是如此,只有当父加载器反馈自己无法完成这个加载请求(它的搜索范围中没有找到所需的类)时,子加载器才会尝试自己去完成加载。

2.5双亲委派模型的作用

  使用双亲委派模型可以让Java中的类随着它的类加载器一起具备一种带有优先级的层次关系,通俗的说:能够保证在各种类加载器环境中是同一个类,因为最终都是委派给最顶端的启动类加载器进行加载。还有就是保证java程序的稳定运作。

3.破坏双亲委派模型

  第二次破坏:JNDI的代码由启动类加载器来完成加载,当时它需要调用其他厂商实现并部署在应用程序的ClassPath下的JNDI服务提供者接口的代码,这就导致了基础类型又要调用回用户的代码。最后Java的设计团队引入了线程上下文类加载器这个不太优雅的设计。

  JNDI服务使用这个线程上下文类加载器去加载所需的SPI服务代码,这是一种父类加载器去请求子类加载器完成类加载的行为,这种行为实际上是打通了双亲委派模型的层次结构来逆向使用类加载器,已经违背了双亲委派模型的一般性原则

  第三次破坏:在代码热替换,模块热部署中,也对双亲委派模型产生了破坏。OSGi是模块化规范一系列规范,它实现模块化热部署的关键是它自定义的类加载器机制的实现。
  每一个程序模块都有一个自己的类加载器,但需要更换一个模块时,就把这个模块连同它的类加载器一起换掉以实现代码的热替换。在加载的过程中,类加载器不再使用双亲委派模型的树状结构,而是更加复杂的网状结构。

  • 将以java.*开头的类,委派给父类加载器加载。
  • 否则,将委派列表名单内的类,委派给父类加载器加载。
  • 否则,将Import列表中的类,委派给Export这个类的Bundle的类加载器加载。
  • 否则,查找当前Bundle的ClassPath,使用自己的类加载器加载。
  • 否则,查找类是否在自己的Fragment Bundle中,如果在,则委派给Fragment Bundle的类加载器加载。
  • 否则,查找Dynamic Import列表的Bundle,委派给对应Bundle的类加载器加载。
  • 否则,类查找失败。
      上面的查找顺序中只有开头两点仍然符合双亲委派模型的原则,其余的类查找都是在平级的类加载器中进行的。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
JVM的类加载是由类加载器及其子类实现的。类加载器Java运行时系统的重要组成部分,负责在运行时查找和加载类文件中的类。在JVM中,类加载器按照一定的层次结构进行组织,每个类加载器负责加载特定位置的类。其中,启动类加载器(Bootstrap ClassLoader)是负责加载存放在<JAVA_HOME>/lib目录中的核心类库,如rt.jar、resources.jar等,同时也可以加载通过-Xbootclasspath参数指定的路径中的类库。启动类加载器是用C语言编写的,随着JVM启动而加载。当JVM需要使用某个类时,它会通过类加载器查找并加载这个类。加载过程会经历连接阶段,包括验证、准备和解析。在验证阶段,JVM会确保加载的类信息符合JVM规范。在准备阶段,JVM会为类变量分配内存并设置初始值,在方法区中分配这些内存。在解析阶段,JVM会根据符号引用替换为直接引用,以便后续的使用。通过类加载器的协同工作,JVM能够在运行时动态加载类,从而实现Java的灵活性和跨平台性。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [JVM 的类加载原理](https://blog.csdn.net/ChineseSoftware/article/details/119212339)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [JVM类加载器](https://blog.csdn.net/rockvine/article/details/124825354)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值