类加载器用以实现类加载功能,并且确定被加载的类在Java虚拟机中的唯一性。
注意:由同一个类加载器加载,则这两个类相等,由不同的类加载器加载,则这两个类不相等。
我们知道类加载器主要分为:
- 启动类加载器
- 扩展类加载器
- 应用程序类加载器
启动类加载器
负责加载以下类:
- 存放在JAVA_HOME\lib目录中的类;
- 被-Xbootclasspath参数所指定路径中、并且是被虚拟机识别的类库。
注意:启动类加载器不能直接被java程序直接饮用。
扩展类加载器
负责加载以下类:
- JAVA_HOME\lib\ext目录中的类;
- 被java.ext.dirs系统变量所指定的路径中的所有类。
注意:开发者可以直接使用扩展类加载器。
应用程序类加载器
负责加载用户类路径(classpath)上所指定的类库。
注意:该类加载器也被称为系统类加载器,开发者可以直接使用该类加载器。若开发者没有自定义类加载器,程序默认使用该类加载器。
各种类加载器并不是孤立的,而是互相配合使用。
在java虚拟机中,各种类加载器配合使用的模型就是双亲委派模型。
扫描二维码关注公众号,回复:
5243762 查看本文章
双亲委派模型
双亲委派模型的工作流程全部在ClassLoader的loadClass()方法中执行:
@Override
protected Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException {
Class<?> c = findLoadedClass(name);
// 检查需要加载的类是否已经被加载过
if (c == null) {
try {
// 若没有加载,则调用父加载器的loadClass()方法
if (parent != null) {
c = parent.loadClass(name, false);
}else{
// 若父类加载器为空,则默认使用启动类加载器作为父加载器
c=findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {
// 若父类加载器加载失败会抛出ClassNotFoundException,
//说明父类加载器无法完成加载请求
}
if(c==null){
// 在父类加载器无法加载时
// 再调用本身的findClass方法进行类加载
c=findClass(name);
}
}
if(resolve){
resolveClass(c);
}
return c;
}
复制代码
若一个类加载器收到了类加载请求:
步骤:
- 把该类加载请求委派给父加载器父完成,而不是自己去加载;(每层的类加载器都是如此,因此所有的类加载请求最终都会交由启动类加载器去加载)
- 只用当父类加载器反馈自己无法完成该加载请求时,自加载器才会自己加载。
优点:
Java类随着它的类加载器一起具备了一种带优先级的层次关系。