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!");
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
3.沙箱安全机制
更多JVM文章请参考我的JVM专栏:https://blog.csdn.net/u011069294/category_10113093.html