类装载器的工作机制
类装载器就是寻找类的字节码文件并构造出类在JVM内部表示对象的组件,在Java中,类装载器把一个类装入JVM中,需要经过以下步骤:
- 装载:查找和导入Class文件
- 链接:执行校验、准备和解析,其中解析是可以选择的
- 校验:检查载入Class文件数据的正确性
- 准备:给类的静态变量分配存储空间
- 解析:将符号引用转换成直接引用
- 初始化:对类的静态变量、静态代码块执行初始化工作
类装载工作由ClassLoader
及其子类负责。ClassLoader
是一个重要的Java运行时系统组件,它负责在运行时查找和装入Class字节码文件。JVM在运行时会产生3个ClassLoader
:根装载器、ExtClassLoader
(扩展类装载器)、AppClassLoader
(应用类装载器)。其中,根装载器不是ClassLoader
的子类,它使用C++语言编写,因而在Java中看不到它,根装载器负责装载JRE的核心类库,如JRE目录下的rt.jar、charsets.jar等。ExtClassLoader
和AppClassLoader
都是ClassLoader
的子类,其中ExtClassLoader
负责装载JRE扩展目录ext
中的JAR类包;AppClassLoader
负责装载Classpath
路径下的类包。
这三个类装载器之间存在父子层级关系,即根装载器是ExtClassLoader
的父装载器,ExtClassLoader
是AppClassLoader
的父装载器。在默认情况下,使用AppClassLoader
装载应用程序的类。
public class ClassLoaderTest {
public static void main(String[] args) {
ClassLoader loader = Thread.currentThread().getContextClassLoader();
System.out.println("current loader: " + loader);
System.out.println("parent loader: " + loader.getParent());
System.out.println("grandparent loader: " + loader.getParent().getParent());
}
}
运行以上代码,在控制台将打印出以下信息:
current loader: sun.misc.Launcher$AppClassLoader@18b4aac2
parent loader: sun.misc.Launcher$ExtClassLoader@6f94fa3e
grandparent loader: null
JVM装载类时使用全盘负责委托机制,全盘负责是指当一个ClassLoader
装载一个类时,除非显示地使用另一个ClassLoader
,该类所依赖及引用的类也由这个ClassLoader
载入;委托机制是指先委托父装载器寻找目标类,只有在找不到的情况下才从自己的类路径中查找并装载目标类。这一点是从安全角度考虑的,java.lang.String
永远是由根装载器来装载的。
ClassLoader的重要方法
-
Class loadClass(String name)
参数name指定类装载器需要装载类的名字,必须使用全限定类名,如
com.smart.beans.Car
-
Class defineClass(String name,byte[] b,int off,int len)
将类文件的字节数组转换成JVM内部的Class对象,字节数组可以从本地文件系统、远程网络获取。参数name为字节数组的全限定类名。
-
Class findSystemClass(String name)
:JVM默认使用的装载机制从本地文件系统导入Class文件。如果本地文件系统不存在该Class文件,则将会抛出
ClassNotFoundException
异常。 -
Class findLoadedClass(String name)
调用该方法来查看
ClassLoader
是否已装入某个类。如果已装入,那么返回java.lang.Class
对象,否则返回null。如果强行装载已存在的类,那么将会抛出链接错误。 -
Class getParent()
获取类装载器的父装载器。除根装载器外,所有的类装载器有且仅有一个父装载器。
ExtClassLoader
的父装载器是根装载器,因为根装载器非Java语言编写,所以无法获得,返回null。
类文件被装载并解析后,在JVM内将拥有一个对应的java.lang.Class
类描述对象,该类的实例都拥有指向这个类描述对象的引用,而类描述对象又拥有指向关联Class Loader
的引用。
每个类在JVM中都拥有一个对应的java.lang.Class
对象,它提供了类结构信息的描述。数组、枚举、注解及基本数据类型,甚至void都拥有对应的Class对象。Class没有public的构造方法。Class对象是在装载类时由JVM通过调用类装载器中的defineClass()
方法自动构造的。