双亲委派机制
双亲委派机制是指当一个类加载器收到一个类加载请求时,该类加载器首先会把请求委派给父类加载器。每一个类加载器都是如此(递归的去查找),只有在父类加载器在本身的搜索范围内找不到指定类时,子类加载器才会尝试本身去加载。
显然,在介绍双亲委派机制的时候,不得不提ClassLoader。再说ClassLoader以前,咱们得先了解下Java的基本知识。
Java是运行在Java的虚拟机(JVM)中的,可是它是怎么就运行在JVM中了呢?咱们在IDE中编写的Java源代码被编译器编译成.class
的字节码文件。而后由咱们的ClassLoader
负责将这些class文件加载到JVM
中去执行。
JVM中提供了三层的ClassLoader:java
暖冬回血~~~~bootstrap
你们好,我是AIO生活,关注我,后续连载更多技术重难点,文章有不足之处,欢迎你们留言指正,谢谢你们啦!安全
顺便提一下,帮你们整理了些JAVA电子书,扫码回复“1024”获取电子书合集,也可签到领现金红包。ssh
Bootstrap ClassLoader(启动类加载器):主要负责加载核心的类库(java.lang.*等),构造Extension ClassLoader
和Application ClassLoader
。函数
Extension ClassLoader(扩展类加载器):主要负责加载jre/lib/ext
目录下的一些扩展的jar。spa
Application ClassLoader(应用程序类加载器):主要负责加载应用程序
的主函数类。code
最后一个CustomClassLoader(用户自定义类加载器) 是java
编写,用户自定义的类加载器,可加载指定路径的class文件
。
对象
那若是有一个Hello.class
文件是如何被加载到JVM中的呢?
咱们简单看一下源码blog
protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException { synchronized (getClassLoadingLock(name)) { // 首先检查这个classsh是否已经加载过了 Class<?> c = findLoadedClass(name); if (c == null) { long t0 = System.nanoTime(); try { // c==null表示没有加载,若是有父类的加载器则让父类加载器加载 if (parent != null) { c = parent.loadClass(name, false); } else { //若是父类的加载器为空 则说明递归到bootStrapClassloader了 //bootStrapClassloader比较特殊没法经过get获取 c = findBootstrapClassOrNull(name); } } catch (ClassNotFoundException e) {} if (c == null) { //若是bootstrapClassLoader 仍然没有加载过,则递归回来,尝试本身去加载class long t1 = System.nanoTime(); c = findClass(name); sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0); sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1); sun.misc.PerfCounter.getFindClasses().increment(); } } if (resolve) { resolveClass(c); } return c; } }
上述这段代码已经很好的解释了双亲委派机制,为了更容易理解,咱们先参看下面描述上述代码流程的图再谈Hello.class
如何加载:
从上图中咱们就更容易理解了,当一个Hello.class
这样的文件要被加载时。不考虑咱们自定义类加载器,首先会在AppClassLoader中检查是否加载过,若是有那就无需再加载了。若是没有,那么会拿到父加载器,而后调用父加载器的loadClass方法。父类中同理会先检查本身是否已经加载过,若是没有再往上。注意这个过程,直到到达Bootstrap ClassLoader以前,都是没有哪一个加载器本身选择加载的。若是父加载器没法加载,会下沉到子加载器去加载,一直到最底层(其实就是递归查找过程),若是没有任何加载器能加载,就会抛出ClassNotFoundException
。递归
因而,咱们就能够很好的总结双亲委派机制的工做流程了:
- 一、当
Application ClassLoader
收到一个类加载请求时,他首先不会本身去尝试加载这个类,而是将这个请求委派给父类加载器Extension ClassLoader
去完成。 - 二、当
Extension ClassLoader
收到一个类加载请求时,他首先也不会本身去尝试加载这个类,而是将请求委派给父类加载器Bootstrap ClassLoader
去完成。 - 三、若是
Bootstrap ClassLoader
加载失败(在<JAVA_HOME>\lib中未找到所需类),就会让Extension ClassLoader
尝试加载。 - 四、若是
Extension ClassLoader
也加载失败,就会使用Application ClassLoader
加载。 - 五、若是
Application ClassLoader
也加载失败,就会使用Custom ClassLoader
(用户自定义加载器)去尝试加载。 - 六、若是均加载失败,就会抛出
ClassNotFoundException
异常。
双亲委派机制的做用
一、防止重复加载同一个.class
。经过委托去向上面问一问,加载过了,就不用再加载一遍。保证数据安全。
二、保证核心.class
不能被篡改。经过委托方式,不会去篡改核心.class
,即便篡改也不会去加载,即便加载也不会是同一个.class
对象了。不一样的加载器加载同一个.class
也不是同一个Class对象。这样保证了Class执行安全。
举个栗子:若是有人想替换系统级别的类:String.java
。篡改它的实现,可是在这种机制下这些系统的类已经被Bootstrap ClassLoader
加载过了,因此并不会再去加载,从必定程度上防止了危险代码的植入。
你们好,我是AIO生活,关注我,后续连载更多技术重难点,文章有不足之处,欢迎你们留言指正,谢谢你们啦!
顺便提一下,帮你们整理了些JAVA电子书,扫码回复“1024”获取电子书合集,也可签到领现金红包。