类的加载过程(生命周期)
一、装载:通过一个类的全限定名获取定义此类的二进制字节流将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构在内存中生成一个代表这个类的java.lang.Class对象(将字节码加载到内存中),作为方法区这个类的各种数据的访问入口 注意:数组类是如何创建加载的呢?
二、链接:验证(Verify):目的在于确保Class文件的字节流中包含信息符合当前虚拟机要求,保证被加载类的正确性,不会危害虚拟机自身安全。主要包括四种验证,文件格式验证,元数据验证,字节码验证,符号引用验证。
准备(Prepare):为类变量分配内存并且设置该类变量的默认初始值,即零值。这里不包含用final修饰的static,因为final在编译的时候就会分配了,准备阶段会显式初始化;这里不会为实例变量分配初始化,类变量会分配在方法区中,而实例变量是会随着对象一起分配到Java堆中。
解析(Resolve):将常量池内的符号引用转换为直接引用的过程。事实上,解析操作往往会伴随着JVM在执行完初始化之后再执行。符号引用就是一组符号来描述所引用的目标。符号引用的字面量形式明确定义在《java虚拟机规范》的Class文件格式中。在解析阶段,jvm根据字符串的内容找到内存区域中相应的地址,然后把符号引用替换成直接指向目标的指针、句柄、偏移量等,这些直接指向目标的指针、句柄、偏移量就被成为直接引用。解析动作主要针对类或接口、字段、类方法、接口方法、方法类型等。对应常量池中的CONSTANT_Class_info、CONSTANT_Fieldref_info、CONSTANT_Methodref_info等。
三、初始化:初始化阶段就是执行类构造器方法(),即clinit方法的过程。此方法不需定义,是javac编译器自动收集类中的所有类变量的赋值动作和静态代码块中的语句合并而来。构造器方法中指令按语句在源文件中出现的顺序执行。()不同于类的构造器。(关联:构造器是虚拟机视角下的())若该类具有父类,JVM会保证子类的()执行前,父类的()已经执行完毕。虚拟机必须保证一个类的()方法在多线程下被同步加锁。到了初始化阶段,才真正开始执行类中定义的Java程序代码,为类的静态变量赋予正确的初始值。
四、使用using
五、卸载unloading
<clinit>()调用会导致死锁么?
对于<clinit>方法的调用,也就是类的初始化,虚拟机在内部会确保其多线程环境中的安全性,所以在执行clinit时会加锁同步,所以clinit是线程安全的,如果在执行clinit时有耗时操作,可能造成线程阻塞,引发死锁。
什么时候会触发的类的加载?
类的加载分为三个阶段:装载loading+链接linking+初始化initialing,链接阶段又可分为验证Verify+准备Prepare+解析Resolve三个子阶段。这个问题实际问的是上述三个阶段都执行的情况,也就是主动使用的情况。Class.forName()会执行类的加载过程,而Class.getClassLoader().loadClass仅仅会执行类的状态load阶段。
类的初始化情况:主动使用vs被动使用
Java程序对类的使用分为两种:主动使用和被动使用
主动使用:Class只有在必须要首次使用的时候才会被装载,java虚拟机不会无条件的状态Class类型,Java虚拟机规定,一个类或接口在初次使用前,必须要进行初始化,这里指的使用,是指主动使用。
被动使用不会执行类的初始化,意味着没有clinit的调用