本文转载自 面试官:请你谈谈Java的类加载过程
一个Java文件从编码完成到最终执行,一般包括两个过程
- 编译
- 运行
编译:把我们写好的java文件,通过javac命令编译成字节码,也就是我们常说的.class文件
运行:则把编译生成.class文件交由Java虚拟机(JVM)执行。
而我们所说的类加载过程就是指JVM把.class文件中类信息加载到内存中,并进行解析生成对应的class对象的过程。
举个通俗的例子来说,JVM在执行某段代码时,遇到了class A,然后此时内存中并没有class A 的相关信息,于是JVM就会到相应的class文件中去寻找class A的类信息,并加载到内存中,这就是我们所说的类加载过程。
由此可见,JVM不是一开始就把所有的类加载到内存中,而是只有第一次遇到某个需要运行的类才会加载。
类加载
- 加载
- 连接
- 初始化
而连接又可以细分为三个小部分 - 验证
- 准备
- 解析
加载
简单来说,加载指的是把class字节码文件从各个来源通过类加载器装载入内存中。
这里有两个重点:
- 字节码来源。一般的加载来源包括从本地路径下编译生成的.class文件,从jar包中.class文件,从远程网络,以及动态代理实时编译
- 类加载器。一般包括启动类加载器,扩展类加载器,应用类加载器,以及用户的自定义类加载器。
注:为什么会自定义类加载器?
- 一方面是由于Java代码很容易被反编译,如果需要对自己的代码加密的话,可以对编译后的代码进行加密,然后再通过实现自己的自定义加载器进行解密,最后再加载。
- 另一方面也有可能可能从非标准的来源加载代码,比如从网络来源,那就需要自己实现一个类加载器,从指定源进行加载。
验证
主要是为了保证加载进来的字节流符合虚拟机规范,不会造成安全错误。
包括对于文件格式的验证,比如常量中是否有不被支持的常量,文件中是否有不规范的或者附加的其他信息?
对于元数据的验证,比如该类是否继承了被final修饰的类?类中的字段,方法是否与父类冲突?是否出现了不合理的重载?
对于字节码的验证,保证程序语义的合理性,比如要保证类型转换的合理性。
对于符号引用的验证,比如校验符号引用中通过全限定名是否能够找到对应的类,校验符号引用中的访问性(private,public等)是否被当前类访问。
准备
主要是为类变量(注意,不是实例变量)分配内存,并且赋予初值。
特别需要注意,初值,不是代码中具体些的初始化的值,而是java虚拟机根据不同变量类型的默认初始值
比如8种引用类型,默认为0;引用类型的初值为null;常量的初值为代码中设置的值。
解析
将常量池内的符号引用替换为直接引用的过程。
两个重点:
- 符号引用 即一个字符串,但是这个字符串给出了一些能够唯一性识别一个方法,一个类的相关信息
- 直接引用 可以理解为一个内存地址,或者一个偏移量。比如类方法,类变量的直接引用是指向方法区的指针,而实例方法,实例变量的直接引用则是从实例的头指针开始算起到这个实例变量位置的偏移量。
举个例子:现在调用方法hello(),这个方法的地址是1234567,那么hello就是符号引用,1234567就是直接引用。
在解析阶段,虚拟机会把所有的类名,方法名,字段名这么符号引用替换为具体的内存地址或偏移量,也就是直接引用。
初始化
这个阶段主要是对类变量的初始化,是执行类构造器的过程。
换句话说,只对static修饰的变量或语句进行初始化。
如果初始化一个类的时候,其父类尚未初始化,则优先初始化其父类。
如果同时包含多个静态变量和静态代码块,则按照自上而下的顺序依次执行。
总结
类加载过程只是一个类声明周期的一部分,在其前,有编译的过程,只有对源代码编译之后,才能获得能够被虚拟机加载的字节码文件;在其后还有具体的类使用过程,当使用完成之后,还会在方法区垃圾回收的过程中进行卸载。
类加载器
类加载器是用来加载Java类到java虚拟机中,一般来说,java虚拟机使用java类的方式如下:java源程序(.java文件)在经历java编译器之后就转为java字节代码(.class文件)。类加载器负责读取java字节码,并转换成java.lang.Class类的一个实例。每个这样的实例用来表示一个java类,通过此实例的newInstance()方法就可以创建该类的一个对象。
类加载器应用在很多方面,比如类层次划分,OSGI,热部署,代码加密等领域。
基本上所有的类加载器都是Java.lang.classLoader类的一个实例。
java.lang.ClassLoader类
java.lang.ClassLoader类的基本职责就是根据一个指定的类的名称,找到或者生成其对应的字节码,然后从这些字节码中定义出一个java类,即java.lang.class类的一个实例。除此之外,ClassLoader还负责加载Java应用所需的资源,如图像文件和配置文件等。
类与类加载器
类加载器虽然只用于实现类的加载动作,但它在java程序中起到的作用却远远不限于类加载阶段。对于任意一个类,都需要由加载它的类加载器和这个类本身一起确立其在java虚拟机中的唯一性。
类加载的种类
- 启动类加载器: 负责加载JRE的核心类库,如jre目标下的rt.jar,charsets.jar等
- 扩展类加载器:负责加载JRE扩展目录ext中的jar类包
- 系统类加载器:负责加载classpath路径下的类包
- 用户自定义加载器:负责加载用户自定义路径下的类包
双亲委派模型
如果一个类加载器收到了类加载请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的类加载器都是如此,因此所有的加载请求最终都应该传送到顶层的启动类加载器中,只有当父类加载器反馈给自己无法完成这个加载请求时(它的搜索范围中没有找到所需的类)时,子加载器才会尝试自己去加载。