类加载器
功能
类加载器负责将未被加载
的.class文件加载到内存,并为之生成对应的java.lang.Class对象。这意味着被加载的类具有唯一性
。
- 类的唯一性:一个类,由其权限定类名和类加载器作为唯一标识。也就是说,不同加载器加载的同一个类被认为是完全不同、互不兼容的。
分类
- 启动类加载器(根类加载器):
使用C++
实现(这里仅限于Hotspot,也就是JDK1.5之后默认的虚拟机),是虚拟机自身的一部分
。它负责加载存放在JDK\jre\lib下,或被-Xbootclasspath参数指定的路径中的,并且能被虚拟机识别的类库。无法被Java程序直接引用
。 - 扩展类加载器:
使用java
实现,该加载器由sun.misc.Launcher$ExtClassLoader实现,它负责加载JDK\jre\lib\ext目录中,或者由java.ext.dirs系统变量指定的路径中的所有类库(如javax.*开头的类),开发者可以直接使用
。 - 应用程序类加载器:
使用java
实现,该类加载器由sun.misc.Launcher$AppClassLoader来实现,它负责加载用户类路径
(ClassPath)所指定的类,开发者可以直接使用
,如果应用程序中没有自定义过自己的类加载器,一般情况下这个就是程序中默认的类加载器。
除了根类加载器器之外,所有类加载器都是ClassLoader
子类的实例。开发者可以通过扩展ClassLoader子类并重写其方法来实现自定义的类加载器。
类加载器的层次关系
这几种类加载器的层次关系如下图所示:
- 这种层次关系称为类加载器的
双亲委派模型
。该模型在JDK1.2期间被引入并广泛应用于之后几乎所有的Java程序中,但它并不是一个强制性的约束模型,而是Java设计者们推荐
给开发者的一种类的加载器实现方式。 - 在双亲委派模型中,除了引导类加载器之外,所有的类加载器都有一个
父类加载器
。因为类加载器 Java 类如同其它的 Java 类一样,也是要由类加载器来加载的。也就是说,加载之间的父子关系并不是继承
关系来实现的,而是类加载器和其加载的实例的关系。一般来说,开发人员编写的类加载器的父类加载器是应用程序类加载器。 - 双亲委派模型的工作流程是:如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把请求委托给父加载器去完成,依次向上,因此,所有的类加载请求最终都应该被传递到顶层的启动类加载器中,只有当父加载器在它的搜索范围中没有找到所需的类时,即无法完成该加载,子加载器才会尝试自己去加载该类。
参考文献
https://www.ibm.com/developerworks/cn/java/j-lo-classloader/index.html 深入探讨Java类加载器
https://blog.youkuaiyun.com/justloveyou_/article/details/72217806 深入理解Java类加载器