类的生命周期2
java类的生命周期就是指一个class文件从加载到卸载的全过程。
完整的生命周期分为7个部分:
装载——验证——准备——解析——初始化——使用——卸载
其中,验证,准备,解析 称之为链接阶段,除了解析以外,其他阶段是顺序发生的,而解析可以与这些阶段交叉进行,因为Java支持动态绑定,需要运行时才能缺点具体内容;在使用阶段实例化对象。
类的初始化
类的加载机制没有明确的触发条件,但是有5种情况必须对类进行初始化,所以 加载--验证--准备 就必须在此之前完成了
类的实例化执行顺序:
父类静态变量(9)——父类静态代码块(8)——子类静态变量(6)——子类静态代码块(5)——父类非静态变量(4)——父类构造函数(2)——子类非静态变量(1)——子类构造函数(0)
静态优先+5,父类优先+3,变量优先+1
类的主动引用
1.通过首次new实例化对象,读取或设置类的静态变量,调用类的静态方法。
2.通过反射指向以上三种行为
3.初始化子类时会触发父类的初始化
4.虚拟机启动时,初始化一个执行主类(作为程序入口直接运行(也就是直接调用main方法)。)
5、使用jdk1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic、REF_putStatic、RE_invokeStatic的方法句柄,并且这个方法句柄对应的类没有进行初始化,则需要先触发其初始化。
注意,有且只有五种情况必须对类进行初始化,这五种情况被称为“主动引用”,除了这五种情况,所有其他的类引用方式都不会触发类初始化,被称为“被动引用”。
import java.lang.reflect.Field;
import java.lang.reflect.Method;
class InitClass{
static {
System.out.println("初始化InitClass");
}
public static String a = null;
public static void method(){}
}
class SubInitClass extends InitClass{}
public class Test1 {
/**
* 主动引用引起类的初始化四:就是运行Test1的main方法时
* 导致Test1初始化,这一点很好理解,就不特别演示了。
* 本代码演示了前三种情况,以下代码都会引起InitClass的初始化,
* 但由于初始化只会进行一次,运行时请将注解去掉,依次运行查看结果。
* @param args
* @throws Exception
*/
public static void main(String[] args) throws Exception{
// 主动引用引起类的初始化一: new对象、读取或设置类的静态变量、调用类的静态方法。
// new InitClass();
// InitClass.a = "";
// String a = InitClass.a;
// InitClass.method();
// 主动引用引起类的初始化二:通过反射实例化对象、读取或设置类的静态变量、调用类的静态方法。
// Class cls = InitClass.class;
// cls.newInstance();
// Field f = cls.getDeclaredField("a");
// f.get(null);
// f.set(null, "s");
// Method md = cls.getDeclaredMethod("method");
// md.invoke(null, null);
// 主动引用引起类的初始化三:实例化子类,引起父类初始化。
// new SubInitClass();
}
}
类的初始化过程是这样的:按照顺序自上而下运行类中的变量赋值语句和静态语句,如果有父类,则首先按照顺序运行父类中的变量赋值语句和静态语句。
在类的初始化阶段,只会初始化与类相关的静态赋值语句和静态语句,也就是有static关键字修饰的信息,而没有static修饰的赋值语句和执行语句在实例化对象的时候才会运行。
被动引用
- 引用父类的静态字段,只会引起父类的初始化,而不会引起子类的初始化。
- 定义类数组,不会引起类的初始化。
- 引用类的常量,不会引起类的初始化。
总结一下使用阶段:
使用阶段包括主动引用和被动引用,主动引用会引起类的初始化(main,初始化子类,new,反射),而被动引用不会引起类的初始化(引用父类的静态资源,定义类数组,引用类常量)。
当使用阶段完成之后,java类就进入了卸载阶段。
卸载
卸载要满足以下条件
- 该类的实列都已被回收,Java对中不存在该类的任何实例
- 加载该类的ClassLoader已被回收
- 该类的对象没有被任何地方引用,无法在任何地方通过反射访问该类的内容
如果以上三个条件全部满足,jvm就会在方法区垃圾回收的时候对类进行卸载,类的卸载过程其实就是在方法区中清空类信息,java类的整个生命周期就结束了。
总结
对象基本上都是在jvm的堆区中创建,在创建对象之前,会触发类加载(加载、连接、初始化),当类初始化完成后,根据类信息在堆区中实例化类对象,初始化非静态变量、非静态代码以及默认构造方法,当对象使用完之后会在合适的时候被jvm垃圾收集器回收。读完本文后我们知道,对象的生命周期只是类的生命周期中使用阶段的主动引用的一种情况(即实例化类对象)。而类的整个生命周期则要比对象的生命周期长的多。