Classloader的双亲委托机制

  • JVM自带的ClassLoader类

    JDK中提供了三个ClassLoader,根据层级从高到低为:

    1. Bootstrap ClassLoader,主要加载JVM自身工作需要的类。
    2. Extension ClassLoader,主要加载%JAVA_HOME%\lib\ext目录下的库类。
    3. Application ClassLoader,主要加载Classpath指定的库类,一般情况下这是程序中的默认类加载器,也是ClassLoader.getSystemClassLoader() 的返回值。(这里的Classpath默认指的是环境变量中配置的Classpath,但是可以在执行Java命令的时候使用-cp 参数来修改当前程序使用的Classpath)

这里写图片描述


  • 什么是双亲委托模型
    JVM加载类的实现方式,我们称为 双亲委托模型

    如果一个类加载器收到了类加载的请求,他首先不会自己去尝试加载这个类,而是把这个请求委托给自己的父加载器,每一层的类加载器都是如此,因此所有的类加载请求最终都应该传送到顶层的Bootstrap ClassLoader中,只有当父加载器反馈自己无法完成加载请求时,子加载器才会尝试自己加载。


  • 双亲委托模型加载类的过程
    我们用一张简单的图片来描述 “使用自定义ClassLoader加载一个类的过程”:

这里写图片描述

  1. 自定义ClassLoader向自己的上层(Application ClassLoader)请求
  2. Application ClassLoader继续向上层(Extension ClassLoader)请求
  3. Extension ClassLoader继续向上层(Bootstrap ClassLoader)请求
  4. Bootstrap ClassLoader是最上层类加载器,所以它尝试在自己的路径中查找要加载类,如果查找到了就加载类,否则向下层(Extension ClassLoader)反馈自己无法加载类。
  5. Extension ClassLoader从自己的路径中寻找要加载类,找到则加载,找不到则继续向下返回。
  6. Application ClassLoader从自己的路径中寻找要加载类,找到则加载,找不到则继续向下返回。
  7. 自定义ClassLoader从自己的路径中寻找要加载类,找到则加载。由于类加载请求是自定义ClassLoader发起的,所以当它自己也找不到要加载的类时会终止加载并抛出
    ClassNotFoundException

  • 为什么要用双亲委托模型
    双亲委托模型的重要用途是为了解决类载入过程中的安全性问题。

    假设有一个开发者自己编写了一个名为java.lang.Object的类,想借此欺骗JVM。现在他要使用自定义ClassLoader来加载自己编写的java.lang.Object类。然而幸运的是,双亲委托模型不会让他成功。因为JVM会优先在Bootstrap ClassLoader的路径下找到java.lang.Object类,并载入它。


  • Java的类加载是否一定遵循双亲委托模型

    这个答案是否定的。
    双亲委托模型只是JDK提供的ClassLoader类的实现方式。
    在实际开发中,我们可以通过自定义ClassLoader,并重写父类的loadClass方法,来打破这一机制。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值