类加载过程
加载–>验证–>准备–>解析–>初始化–>使用–>卸载
开始顺序是如上所示,但并非串联,两个阶段部分之间可同时运行。如加载和验证
加载
1、通过全限定名获取此类的二进制字节流。
2、将该二进制字节流代表的静态存储结构转化为方法区运行时的数据结构。
3、生成该类的java.lang.Class,作为访问入口。
验证
1、文件格式验证,字节流是否符合Class文件格式
2、元数据验证,字节码描述的类信息是否符合Java语法
3、字节码验证,确定程序语义(方法体)不会危害虚拟机
4、符号引用验证,在解析阶段发生,确保解析顺利执行。
准备
为类变量(即静态变量)初始化,赋0或false。静态常量初始化则直接赋值对应的数据
解析
将常量池符号引用转换成直接引用。主要针对:类、接口、字段、类方法、接口方法、方法类型、方法句柄和调用点限定符。符号引用可以理解为内存中的逻辑地址,直接引用类似内存中物理地址,与虚拟机内存布局有关。
初始化
初始化阶段时执行构造器clinit()的过程。按照顺序收集变量的赋值动作和静态语句块。类中父类的clinit方法会先于子类执行;而接口中却不会,除非父接口的变量(也就是静态常量)被使用。虚拟机会保证一个类能在多线程中仅执行一次clinit方法。
有以下情况会触发类的初始化(若类还未被加载),称为主动引用:
1、遇到new、getstatic、putstatic、invokestatic这4条字节码指令。Java场景有:new关键字实例化对象、读取或设置类静态字段(被final修饰、已在编译器放入常量池的静态字段除外)、以及调用静态方法。
2、使用反射调用类
3、初始化类时发现其父类还未被初始化,先初始化父类。
4、包含main方法的类
5、java1.7 方法句柄对应的类
被动引用不会触发初始化
1、子类调用父类的静态(static)变量,只会加载父类,而不会加载子类。
2、对于对象数组而言,不会加载对应的类类型,而会触发另一个类类型的加载(是数组类)
3、静态常量由于在编译器已放入静态常量池,则不会触发类的初始化。