目录
static执行顺序
有下面一段代码,你知道它的执行结果吗?
public class StaticDemoSupper {
static {
System.out.println("A");
}
/**
*
*/
public StaticDemoSupper() {
System.out.println("B");
}
}
public class StaticDemoChild extends StaticDemoSupper {
static{
System.out.println("C");
}
/**
*
*/
public StaticDemoChild() {
System.out.println("D");
}
public static void main(String[] args) {
new StaticDemoChild();
}
}
代码的执行结果如下:
A
C
B
D
通过这段代码的执行结果我们可以知道它的执行顺序如下
父类的静态代码块->子类的静态代码块->父类的构造方法->子类的构造方法
static修饰的代码块不属于任何类的对象,它是属于类的;所以静态代码块是随着类加载而加载;
类加载过程
加载 -> 链接(验证,准备,解析)-> 初始化
加载
- 通过一个类的完全限定名获取定义这个类的二进制字节流
- 将这个类的字节流代表的静态存储对象结构转化为方法区运行时数据
- 在堆中生成这个类的class对象
链接
验证
- 文件格式验证
- 元数据验证
- 字节码验证
- 符号引用验证
准备
正式为类分配内存,将变量初始化为“零”,基本数据类型为0 ,对象为null
解析
解析阶段将常量池内的符号引用转化为直接引用
初始化
初始化是类加载的最后一步,而这一步也是真正意义上执行类定义的JAVA代码。
(1) 遇 到new、getstatic、putstatic或invokestatic这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。生成 这4条指令的最常见的Java代码场景是:使用new关键字实例化对象的时候,读取或设置一个类的静态字段(被final修饰、已在编译期把结果放入常量 池的静态字段除外)的时候,以及调用一个类的静态方法的时候。
(2) 使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。
(3) 当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化。
(4)当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类。
只有上述四种情况会触发初始化,也称为对一个类进行主动引用,除此以外,所有其他方式都不会触发初始化,称为被动引用