-
工作原理
- 1、如果一个类加载器收到类加载请求,他并不会自己先去加载,而是把这个请求委托给自己的父类加载器去执行
- 2、 如果父类还存在父类加载器,则继续向上委派,依次递归最终请求将到引导类加载器(bootStrap ClassLoader)
- 3、如果父类加载器能够完成类的加载任务,就成功返回,倘若无法完成,子类加载器就自己去加载,这就是双亲委派模式 优势
- 1、避免类的重复加载
- 2、保护程序安全,防止核心API被随意篡改
例子
比如在同项目下创建了java.lang包下的String,并在一个测试中引用了String类,项目启动并不会去加载自定义下的java.lang包下的String,只会加载jdk下的jre下的包,既使用的是jdk下的String
public class StringTest {
public static void main(String[] args) {
String s = new String();//启动这个的时候并不会有下面的输出,说明并没有加载下面的类
}
}
package java.lang;
public class String {
static {
System.out.println("这是自定义的String!!!");
}
}
报错
在核心api里面没有main方法,所以下面那种方式会报错,说找不到main方法,原因是在加载的时候会由父类加载器去找,最终在引导类中找到了String类,然而String类中并没有main方法,所以报错了,而不是找的该项目下的String
错误: 在类 java.lang.String 中找不到 main 方法, 请将 main 方法定义为:
public static void main(String[] args)
否则 JavaFX 应用程序类必须扩展javafx.application.Application
public class String{
public static void main(String[] args){
System.out.print("这是String的启动类");
}
}
- jvm中判断两个class对象是否为同一类存在两个必要条件;
- 类的完整性必须一致,包括包名
- 加载类的ClassLoader必须相同
面试官会问“介绍一下双亲委派的过程?”
- 1、在类加载的时候,系统会首先判断当前类是否被加载过。已经被加载的类会直接返回,否则才会尝试加载(每个父类加载器都会走一遍这个流程)。
- 2、类加载器在进行类加载的时候,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成(调用父加载器 loadClass()方法来加载类)。这样的话,所有的请求最终都会传送到顶层的启动类加载器 BootstrapClassLoader 中。
- 3、只有当父加载器反馈自己无法完成这个加载请求(它的搜索范围中没有找到所需的类)时,子加载器才会尝试自己去加载(调用自己的 findClass() 方法来加载类)。