Java程序对类的使用方式可分为两种:
- 主动使用
- 被动使用
所有的Java虚拟机实现必须在每个类或接口被Java程序“首次主动使用”时才初始化他们。
主动使用(七种)
- 创建类的实例 new Object()
- 访问某个类或接口的静态变量,或者对该静态变量赋值
- 调用类的静态方法
- 反射(如Class.forName("com.test.Test"))
- 初始化一个类的子类
// 在初始化Child的时候也会初始化Parents
private class Child extends Parents{}
- Java虚拟机启动时被标明为启动类的类(包含main方法的类。例如:Java Test)
- JDK1.7开始提供的动态语言支持:
Java.lang.invoke.MethodHandle 实例的解析结果REF_getStatic,REF_putStatic,REF_invokeStatic句柄对应的类没有初始化则初始化。
除了以上七种情况,其他使用Java类的方式都被看做是对类的被动使用,都不会导致类的初始化。
类的加载:
类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在内存中创建一个java.lang.Class对象(规范并未说明Class对象位于哪里,HotSopt虚拟机将其放在了方法区中)用来封装类在方法区内的数据结构。
加载.class文件的方式:
- 从本地系统中直接加载
- 通过网络下载.class文件
- 从zip,jar等归档文件中加载.class文件
- 从专有数据库中提取.class文件
- 将Java源文件动态编译为.class文件(动态代理和web开发时用到的jsp,最终会被转换为一个servlet,最后转换为class文件)
主动使用例子:
public class MyTest1 {
public static void main(String[] args){
System.out.println(MyChild1.str);
}
}
class MyParent1{
public static String str = "hello world";
static{
System.out.println("MyParent1 static block");
}
}
class MyChild1 extends MyParent1{
static {
System.out.println("MyChild1 static block");
}
}
执行结果:
子类的静态代码块没有执行
将子类代码修改
public class MyTest1 {
public static void main(String[] args){
System.out.println(MyChild1.str2);
}
}
class MyParent1{
public static String str = "hello world";
static{
System.out.println("MyParent1 static block");
}
}
class MyChild1 extends MyParent1{
public static String str2 = "Welcome";
static {
System.out.println("MyChild1 static block");
}
}
执行结果如下,子类的静态代码块得到了执行
第一个例子,对于静态字段来说,只有直接定义了该字段的类才会被初始化,虽然使用了MyChild的名字,但是用的父类的static str 所以不算对MyChild的主动使用
第二个例子父类初始化是因为第五种主动使用的情况,初始化一个类的子类,要求其父类全部都已经初始化完毕