今天给大家分享一道国内的一二线互联网公司,高频次出现的面试题。比如,什么是类加载?什么是双亲委派?等等。首先,我们来看双亲委派,它全称是Parent Delegation Model,直译过来可能叫做父级委托模型更容易理解。不管它叫什么,如果是你被问到这样的问题,你会不会冷场呢?
那么,今天我给大家分享一下我的理解。
1、类加载机制
要理解双亲委派,首先要理解Java的类加载机制。接下来,我简单介绍一下Java的类加载机制,如图所示:
Java编译器将Java源文件编译成.class文件,再由JVM加载.class文件到内存中,JVM装载完成后得到一个Class字节码对象。拿到字节码对象之后,我们就可以实例化了。
那么,类的加载过程需要使用到加载器。JVM设计了3个类加载器,它们分别是:Bootstrap类加载器、Extension类加载器和 类加载器,这些类加载器分别加载不同作用范围的jar包和.class文件。下面,我给大家详细介绍一下:
1、Bootstrap ClassLoader,主要是负责Java核心类库的加载,也就是 %{JDK_HOME}lib下的rt.jar、resources.jar等
2、Extension ClassLoader,主要负责%{JDK_HOME}libext目录下的jar包和class文件
3、Application ClassLoader,主要负责当前应用里面的classpath下的所有jar包和类文件
除了系统自己提供的类加载器以外,还可以通过ClassLoader类实现自定义加载器,去满足一些特殊场景的需求。
2、双亲委派机制
所谓双亲委派机制,或者叫父级委托模型,就是指按照类加载器的层级关系,逐层进行委派。如图所示:
比如,我们需要加载一个class文件的时候,首先会把这个class的查询和加载委派给父加载器去执行,如果父加载器都无法加载,再尝试自己来加载这个class。
3、总结
我认为,双亲委派机制的设计有2个好处:
保证安全性,因为这种层级关系实际上代表的是一种优先级,也就是所有的类的加载,优先给Bootstrap ClassLoader。那对于核心类库中的类,就没办法去破坏,比如自己写一个java.lang.String,最终还是会交给Bootstrap类加载器加载。再加上每个类加载器的都有不同的作用范围,这就意味着自己写的java.lang.String就没办法去覆盖核心类库中类。避免重复加载,我认为这种层级关系的设计,可以避免重复加载导致程序混乱的问题。如果父加载器已经加载过了,那么子加载器就没必要去加载了。
好了,以上就是我对双亲委派机制的理解。