JVM深入理解(一)
类加载
1. 简述
在Java代码中,类型的加载、连接和初始化是在运行期间完成的。这也是Java区别于C/C++这类纯静态语言的重要原因。
2. 详述类的加载、连接和初始化
-
加载:查找并记载类的二进制数据
-
连接:
-
验证:验证字节码文件的正确性
- 准备:为类的静态变量分配内存(此时类还未被创建),并初始化默认值
- 解析:把类中的符号引用,转化为直接引用
-
初始化:为类中的静态变量赋予正确的初始值
-
//例如
public static test{
private static int a = 1;
//这时的a在准备阶段就不是1而是0
//而在初始化的时候才是1
}
Java对类的使用方式分为主动使用和被动使用
所有的Java虚拟机实现必须在每个类或者接口被Java程序首次主动使用时才会初始化他们。
3. 主动使用和被动使用
主动使用
1.创建类的实例
2.访问某个类或者接口的非final类型的静态变量/对该静态变量赋值
3.调用类的静态方法
4.反射
5.初始化一个类的子类
6.jvm启动时被标明为启动类的类(如Java Test、main方法所在的类)
7.jdk1.7开始提供动态语言的支持相关(很少使用)
注:
初始化一个类的子类这条规则,不适用于接口,即:初始化一个类的子类,会先初始化它的父类,但是不一定会初始化它的接口。只有当首次使用该接口的静态变量时,才会初始化
被动使用
1. 通过子类引用父类的静态字段,为子类的被动使用
2. 通过数组定义类引用类,为类的被动使用,不会触发此类的初始化
3.常量在编译阶段会存入调用方法所在的类的常量池中),本质上没有直接引用到定义常量的类,因此不会触发定义常量的类的初始化
详见CSDN博客
例子
public class MyTest01 {
public static void main(String[] args) {
//System.out.println("调用子类输出父类的静态变量 -->" + MyChild.str1);
//System.out.println("调用子类输出子类的静态变量 -->" + MyChild.str2);
}
}
//父类
class MyParent {
static String str1 = "Parent";
static {
System.out.println("父类静态代码块");
}
}
//子类
class MyChild extends MyParent {
static String str2 = "Child";
static {
System.out.println("子类静态代码块");
}
}
输出结果为:
当使用子类输出父类的成员变量str1时,相当于主动使用了父类,被动使用了子类,因此父类的静态代码块被调用了而子类的却没有。
当使用子类输出子类的成员变量str2时,相当于主动使用了子类,但当子类被初始化时,会默认先初始化其父类,相当于主动使用了父类,因此同时输出了子类和父类的静态代码块。