8.JVM双亲委派机制(面试常问)
1.什么是双亲委派机制?
双亲委派机制工作原理:(面试)
1.如果一个类加载器收到了类加载请求,它并不会自己先去加载,而是把这个请求委托给父类的加载器去执行。
2.如果父类的加载器还存在其父类加载器,则进一步向上委托,依次递归,请求最终会到达顶层的启动类加载器。(从这里就可以看出来,类加载请求都会先到达启动类加载器)
3.如果父类加载器可以完成类加载任务,就成功返回,倘若无法完成此加载任务,则委派给它的子加载器去加载。
如图所示,如果有个类加载请求来了,会一直向上委托,直到引导类加载器;然后引导类加载器尝试加载,如果它不能加载,则会给他的子加载器扩展类加载器加载;如果扩展类加载器还是不能加载;则再到下一级系统类加载器。
例子:
程序中需要使用spi接口,所以需要加载spi类库。通过双亲委派机制,加载spi的请求会到达引导类加载器,由于spi接口属于java的核心api,所以引导类加载器会直接进行加载。
spi是接口,要使用它的实现类,就涉及到了第三方的jar包,下图所示的例子使用jdbc类库,需要加载jdbc.jar。
需要加载第三方的jar包,不属于核心api,这时候需要系统类加载器进行加载。这时候会进行反向委派,引导类加载器会委派给扩展类加载器,扩展类加载器会委派给系统类加载器。实际上是由线程上下文加载器加载的,线程上下文加载器是系统类加载器的一种。
这个例子可以看到,接口是由引导类加载器加载的,而具体的实现类是由线程上下文加载器(也就是系统类加载器)加载的。
2.双亲委派机制的优势
1.避免类的重复加载。一旦一个类被父类加载器加载之后,就不会再被委派给子类进行加载。
2.保护程序安全。
保护程序安全的例子:
运行下面的例子,会直接报错。
运行main函数,需要加载ShkStart,根据双亲委派机制,加载请求会被向上委派到引导类加载器(记住第1小节的工作原理图);引导类加载器一看,包是java.lang,所以是由它来进行加载。
加载会直接报错,因为自己定义的类的包名为java.lang。
package java.lang; // 包命名为java.lang /** * @author shkstart * @create 2020 下午 12:00 */ public class ShkStart { public static void main(String[] args) { System.out.println("hello!"); } }