类从加载到虚拟机到卸载,它的整个生命周期包括:加载(Loading),验证(Validation),准备(Preparation),解析(Resolution),初始化(Initialization),使用(Using)和卸载(Unloading)。其中,验证、准备和解析部分被称为连接(Linking)。
加载:
在加载阶段,虚拟机主要完成三件事:
1.通过一个类的全限定名来获取定义此类的二进制字节流。也就是得到class文件的字节流,但是没有指明在那里读取,有可能是代理类,jspapplet,zip中读取的class文件。
2.将这个字节流所代表的静态存储结构转化为方法区域的运行时数据结构。class文件结构就是静态存储结构;前面提到,方法区存在(运行时常量池),存放着类的字面量和符号引用,这部分内容在加载时进入运行时常量池,所以称为运行时数据结构。
3.在内存(不明确是Java堆)中生成一个代表这个类的java.lang.Class对象,作为程序访问方法中这些类型数据的外部接口,可以用这个对象进行反射获得对象信息。
验证:
验证阶段作用是保证Class文件的字节流包含的信息符合JVM规范,不会给JVM造成危害。如果验证失败,就会抛出一个java.lang.VerifyError异常或其子类异常。验证过程分为四个阶段:
1.文件格式验证:验证字节流文件是否符合Class文件格式的规范,并且能被当前虚拟机正确的处理。
2.元数据验证:是对字节码描述的信息进行语义分析,以保证其描述的信息符合Java语言的规范。
3.字节码验证:主要是进行数据流和控制流的分析,保证被校验类的方法在运行时不会危害虚拟机。
4.符号引用验证:符号引用验证发生在虚拟机将符号引用转化为直接引用的时候,这个转化动作将在解析阶段中发生。
准备:
准备阶段为变量分配内存并设置类变量的初始值。在这个阶段分配的仅为类变量(static修饰的变量),而不包括实例变量。对已非final的变量,JVM会将其设置成“零值”,而不是其赋值语句的值:
pirvate static int size = 12;
那么在这个阶段,size的值为0,而不是12。 final修饰的类变量将会赋值成真实的值。因为final修饰的字段在class文件的字段属性表中存在ConstantValue,准备阶段虚拟机会把变量赋值为ConstantValue。
解析:
解析过程是将常量池内的符号引用替换成直接引用。主要包括四种类型引用的解析。类或接口的解析、字段解析、方法解析、接口方法解析。
初始化:
在准备阶段,类变量已经经过一次初始化了,在这个阶段,则是根据程序员通过程序制定的计划去初始化类的变量和其他资源。这些资源有static{}块,构造函数,父类的初始化等。
在编译生成class文件时,会自动生成两个方法 clinit()和init() 方法
- clinit方法是指变量中的赋值动作和static{}块
- 而init方法是指构造方法。
注意:
1.
class Test{
static {
i = 0;//1
System.out.print(i);//2
}
static int i = 1;
}
编译器收集的顺序是语句的顺序,静态语句块只能访问定义在语句块前面的变量,对于后面的变量,只能赋值,但是不能访问,因此//1没问题,//2提示“非法向前引用” 错误。
2.子类的init方法需要显式调用父类的init方法,但是clinit方法不用,子类的clinit方法一定在父类的clinit方法执行完以后。
父类{
public static ini A = 1;
static {
A = 2;
}
}
子类 extend 父类{
public int B = A;
}
main{
syso(子类.B)
}
结果是2,因为父类先执行完Clinit方法。
3.Clinit方法不是必须的,如果没有 变量中的赋值动作和static{}块,那么它没有clinit方法。
4.接口没有static{}语句块,但是可以有变量中的赋值动作,所以它也会有clinit方法,但是与类不同的是,它不会先执行父类的clinit方法。
5.Clinit方法是线程安全的。