说到双亲委派机制,就必须要先要弄清楚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会判断能否完成加载任务,如果能则直接加载,否则会将加载任务交给儿子类加载器;
- 儿子类加载器也会判断能否完成加载任务,如果能则直接加载,否则会再一次将加载任务交给儿子类加载器;
- 不断的循环进行步骤5,直到最后一个类加载器,如果这个类加载器仍然不能够加载这个类,就会抛出一个异常:ClassNotFoundException。
双亲委派机制的好处是什么
- 首先,保证了java核心库的安全性。如果你也写了一个java.lang.String类,那么JVM只会按照上面的顺序加载jdk自带的String类,而不是你写的String类。
- 其次,还能保证同一个类不会被加载多次。