什么是双亲委派机制
了解双亲委派,需要先了解下JAVA的类加载器ClassLoader,java的类加载器主要有以下几种
- BootStrapClassLoader:启动类加载器,该ClassLoader是jvm在启动时创建的,用于加载$JAVA_HOME/jre/lib下面的类库。由于引导类加载器涉及到虚拟机本地实现细节,开发者无法直接获取到启动类加载器的引用,所以不能直接通过引用进行操作。
- ExtClassLoader:扩展类加载器,该ClassLoader会加载 $JAVA_HOME/jre/lib/ext下的类库。
- AppClassLoader:应用程序类加载器,AppClassLoader会加载java环境变量CLASSPATH所指定的路径下的类库,而CLASSPATH所指定的路径可以通过System.getProperty("java.class.path")获取;当然,该变量也可以覆盖,可以使用参数-cp,例如:java -cp 路径 (可以指定要执行的class目录)。
- CustomClassLoader:自定义类加载器,该ClassLoader是指我们自定义的ClassLoader,比如tomcat的StandardClassLoader属于这一类;当然,大部分情况下使用AppClassLoader就足够了。
当我们写了一个java类,然后编译为class,那么这几个类加载器是如何加载class的呢?
- AppClassLoader首先不会自己加载,而是会委派给自己的父加载器ExtClassLoader去完成(先查找自己的缓存,如果存在,则返回,不存在,加载)
- ExtClassLoader同理不会自己加载,而是委派给父加载器BootStrapClassLoader去完成(先查找自己的缓存,如果存在,则返回,不存在,加载)
- 如果BootStrapClassLoader加载失败,也就是没有在自己的加载路径下找到此类,则加载失败,然后会使用ExtClassLoader尝试加载(先查找自己的缓存,如果存在,则返回,不存在,加载)
- 同理如果ExtClassLoader加载失败会交给AppClassLoader,如果AppClassLoader加载失败,则会抛出ClassNotFoundException
双亲委派机制的意义
jdk为了处于安全而考虑,比如你自己写一个java.lang.String类,jvm是不会加载这个类的,因为在启动jvm的时候就加载过了,所以BootStrapClassLoader会直接返回,防止别人破坏