本文参考:
jvm系列(一):java类的加载机制
java类加载时机与过程
java类的加载过程
类从被加载到虚拟机内存中开始,直到卸载出内存为止,它的整个生命周期包括了:加载、验证、准备、解析、初始化、使用和卸载这7个阶段。其中,验证、准备和解析这三个部分统称为连接。
加载:通过类的全限定名获取二进制字节流,解析字节流,将类的结构存储到方法区,在堆中生成一个对应的java.lang.Class对象,通过这个对象访问方法区中的数据。
验证:确保被加载的类的正确性。
准备:为类的静态变量分配内存,并将其初始化为默认值,初始化的是各数据类型的默认值,如0、0L、null、false等,而非代码中赋的初值。
解析:把类中的符号引用转换为直接引用。个人理解就是把代码里语义中包含引用的部分,创建对应的实际引用。
初始化:为类的静态变量赋予正确的初始值。
初始化步骤为:
- 假如这个类还没有被加载和连接,则程序先加载并连接该类
- 假如该类的直接父类还没有被初始化,则先初始化其直接父类
- 假如类中有初始化语句,则系统依次执行这些初始化语句
执行顺序
在这五个阶段中,加载、验证、准备和初始化这四个阶段发生的顺序是确定的,而解析阶段则不一定,它在某些情况下可以在初始化阶段之后开始,这是为了支持Java语言的运行时绑定)。另外注意这里的几个阶段是按顺序开始,而不是按顺序进行或完成,因为这些阶段通常都是互相交叉地混合进行的,通常在一个阶段执行的过程中调用或激活另一个阶段。
执行时机
以上五个步骤中,第一步加载的时机并没强行约束,这点可以交给虚拟机的的具体实现,作为开发者需要着重了解初始化的时机:只有当对类的主动使用的时候才会导致类的初始化,类的主动使用包括以下六种:
-
创建类的实例,也就是new的方式
-
访问某个类或接口的静态变量,或者对该静态变量赋值
-
调用类的静态方法
-
反射(如 Class.forName(“com.shengsiyuan.Test”))
-
初始化某个类的子类,则其父类也会被初始化
-
Java虚拟机启动时被标明为启动类的类( JavaTest),直接使用 java.exe命令来运行某个主类。
以上情况称为称对一个类进行“主动引用”,除此种情况之外,均不会触发类的初始化,称为“被动引用”。
插入一个java面试题
class SingleTon {
private static SingleTon singleTon = new SingleTon();
public static int count1;
public static int count2 = 0;
private SingleTon() {
count1++;
count2++;
}
public static SingleTon getInstance() {
return singleTon;
}
}
public class Test {
public static void main(String[] args) {
SingleTon singleTon = SingleTon.getInstance();
System.out.println("count1=" + singleTon.count1);
System.out.println("count2=" + singleTon.count2);
}
}
知道了整个类的加载过程,就很容易分析。
首先在
SingleTon singleTon = SingleTon.getInstance();
发生了类的主动引用。
此时,会执行SingleTon类的整个加载过程。
准备阶段会把SingleTon的变量count1和count2都赋值为默认值0;
初始化阶段开始,先执行第一句
private static SingleTon singleTon = new SingleTon();
实例化会执行构造函数
private SingleTon() {
count1++;
count2++;
}
此时
count1=1
count2=1
再执行后面的
public static int count1;
public static int count2 = 0;
count1没有赋值操作,count2赋值为0,所以正确答案是
count1=1
count2=0