双亲委派机制,就必须弄清楚Java的类加载器。
什么是类加载器
Java类加载器(ClassLoader)是Java运行时环境(JRE)的一部分,负责动态的将Java类加载到Java虚拟机的内存空间。
类加载器有哪些
主要有三个:
- 引导类加载器(Bootstrap ClassLoader):加载Java的核心库(jre/lib/rt.jar),同时加载另外两种类加载器,由C++编写
- 扩展类加载器(Extensions ClassLoader):加载Java的扩展库(jre/ext/*.jar)
- 应用类加载器(Application ClassLoader):它根据 Java 应用的类路径(CLASSPATH)来加载 Java 类。一般来说,Java 应用的类都是由它来完成加载。
除了上面三种外,还可以自定义类加载器,以方便开发。
类加载器之间的关系
代码示例:
结果输出:
通过这段代码看出,日常编写的代码,是通过Application ClassLoader加载,其父类是Extensions ClassLoader。
实际上,Bootstrap ClassLoader也是Extensions ClassLoader的父类,但Bootstrap ClassLoader由C++编写,在Java中无法获取,所以对外显示为null。
双亲委派机制
双亲委派机制指,当某个特定的类加载器在收到类加载的请求时,会遵循下面的规则顺序:
- 先判断被加载的类是否加载过,如果是则结束,否则会将加载任务委托给自己的父亲;
- 父类加载器在收到类加载的请求时,也先判断被加载的类是否加载过,如果是则结束,否则同样将加载任务委托给自己的父亲
- 不断的循环进行步骤2,直到将加载任务委托给Bootstrap ClassLoader为止。此时,Bootstrap ClassLoader 会先判断被加载的类是否加载过,如果是则结束;
请注意,到这里为止,都只是在转移加载任务的请求,下面将会进行类加载。
- Bootstrap ClassLoader会判断能否完成加载任务,如果能则直接加载,否则会将加载任务交给子类加载器;
- 子类加载器也会判断能否完成加载任务,如果能则直接加载,否则会再一次将加载任务交给子类加载器;
- 不断的循环进行步骤2,直到最后一个类加载器,如果这个类加载器仍然不能够加载这个类,就会抛出一个异常:ClassNotFoundException。
双亲委派机制的好处
- 首先,保证 java核心库 的安全性。如果你也写了一个java.lang.String类,那么JVM只会按照上面的顺序加载jdk自带的String类,而不是你写的String类。
- 其次,保证同一个类不会被加载多次。