jvm的类加载过程实例讲解

1.类的加载机制

类从被加载到虚拟机内存中开始,直到卸载出内存为止,它的整个生命周期包括了:加载、验证、准备、解析、初始化、使用和卸载这7个阶段。其中,验证、准备和解析这三个部分统称为连接(linking)

jvm的类加载过程实例讲解_第1张图片

其中,加载、验证、准备、初始化和卸载这五个阶段的顺序是确定的,类的加载过程必须按照这种顺序按部就班的“开始”(仅仅指的是开始,而非执行或者结束,因为这些阶段通常都是互相交叉的混合进行,通常会在一个阶段执行的过程中调用或者激活另一个阶段),而解析阶段则不一定(它在某些情况下可以在初始化阶段之后再开始,这是为了支持Java语言的运行时绑定。

2.何时开始类的初始化

什么情况下需要开始类加载过程的第一个阶段:"加载"。虚拟机规范中并没强行约束,这点可以交给虚拟机的的具体实现自由把握,但是对于初始化阶段虚拟机规范是严格规定了如下几种情况,如果类未初始化会对类进行初始化。

创建类的实例

访问类的静态变量(除常量【被final修辞的静态变量】原因:常量一种特殊的变量,因为编译器把他们当作值(value)而不是域(field)来对待。如果你的代码中用到了常变量(constant variable),编译器并不会生成字节码来从对象中载入域的值,而是直接把这个值插入到字节码中。这是一种很有用的优化,但是如果你需要改变final域的值那么每一块用到那个域的代码都需要重新编译。

访问类的静态方法

反射如(Class.forName("my.xyz.Test"))

当初始化一个类时,发现其父类还未初始化,则先出发父类的初始化

虚拟机启动时,定义了main()方法的那个类先初始化

以上情况称为称对一个类进行“主动引用”,除此种情况之外,均不会触发类的初始化,称为“被动引用”

接口的加载过程与类的加载过程稍有不同。接口中不能使用static{}块。当一个接口在初始化时,并不要求其父接口全部都完成了初始化,只有真正在使用到父接口时(例如引用接口中定义的常量)才会初始化。

3.被动引用的例子

1.子类调用父类的静态变量,子类不会被初始化。只有父类被初始化。。对于静态字段,只有直接定义这个字段的类才会被初始化.

2.通过数组定义来引用类,不会触发类的初始化

3.访问类的常量,不会初始化类

class SuperClass {  

static {  

System.out.println("superclass init");  

    }  

public static int value = 123;  

}  

class SubClass extends SuperClass {  

static {  

System.out.println("subclass init");  

    }  

}  


public class Test {  

public static void main(String[] args) {  

System.out.println(SubClass.value);// 被动应用1  

SubClass[] sca =new SubClass[10];// 被动引用2  

    }  

}  

程序运行输出    superclass init

                            123

从上面的输入结果证明了被动引用1与被动引用2


class ConstClass {  

static {  

System.out.println("ConstClass init");  

    }  

public static final String HELLOWORLD = "hello world";  

}  


public class Test {  

public static void main(String[] args) {  

System.out.println(ConstClass.HELLOWORLD);// 调用类常量  

    }  

}  

程序输出结果

hello world

从上面的输出结果证明了被动引用3

你可能感兴趣的:(jvm的类加载过程实例讲解)