类的加载过程
加载
- 将 .java 文件转成二进制流
- 将类信息存入方法区
- 在方法区中生成一个 Class 对象,用于访问该类的接口
PS:
读取到二进制流的来源,常见的有:ZAR包,运行中动态计算(动态代理),其它文件生成(JSP)
验证
验证过程是为了防止恶意代码破坏虚拟机而进行验证的过程。
1. 文件格式验证:该过程是针对二进制流进行的,只有通过了该验证,二进制流才可以进去到方法区中。文件格式验证交叉在加载过程中。
2. 元数据验证
3. 字节码验证:针对数据流和控制流的验证
4. 符号引用验证,该阶段发生在解析过程中
准备
准备过程是对类变量、常量设置初值,类变量和静态常量设置的初始值是不同的
private static int a = 12; // 这里面给类变量 a 设置的初始值是 0
private static final int a = 12; // 这里给静态常量 a 设置的初始值是 12
解析
解析是将符号引用转换为直接引用的过程。
初始化
初始化的条件:
1. 遇到 new, putStatic, getStatic, invokeStatic 时对当前类进行初始化
2. 初始化子类时,如果父类没有初始化,则对父类进行初始化
3. 对该类进行反射操作时
4. 含 main() 方法的类在虚拟机启动时,会先初始化这个类
5. 调用该类的静态方法时,如果还没有进行初始化,则对该类进行初始化
不会初始化的情况:调用子类的静态方法,不会初始化父类;调用常量,不会初始化常量所在的类,因为常量会保存在使用类的常量池;
初始化的过程就是执行 <clinit>()的过程。
<clinit>()是生成的类的构造器,由成员变量的赋值语句和静态语句块组合而成,如果没有需要赋值的变量和静态语句块,那么 <clinit>()也可以不存在;
虚拟机会保证在执行 <clinit>()方法之前,其父类的 <clinit>()方法已经执行完毕;
<clinit>()会保证正确的加锁,同步;
接口也同样存在 <clinit>()构造方法,接口执行构造方法之前不必先执行父类的 <clinit>()
类的卸载
类的卸载,即清除类在方法区的信息
类卸载的触发条件:
1. 该类的实例都已被回收
2. 没有对该类反射
3. 加载该类的类加载器都已被回收
类加载器
类加载器和类共同唯一决定一个类在 Java 虚拟机中的唯一性。
三种类加载器
启动类加载器:负责 <JAVA_HOME>\lib 目录中的,无法被 Java 代码直接引用
扩展类加载器:负责 <JAVA_HOME>\lib\ext 目录中的,开发者可以直接使用
程序类加载器:我们平时写的类就默认由它来加载
双亲委派机制
每个类加载器在 loadClass 时先交给父类加载器,一层层网上传递,如果父类加载器处理不了才自己处理。这里的父类加载器是用组合实现的,而不是直接继承。
破坏双亲委派机制
为了实现自定义的类加载器。