类的初始化顺序
将一个类加载到Java虚拟机中需要经历三个阶段:加载->链接(验证、准备,解析)->初始化。
- 加载:这是由类加载器(ClassLoader)执行的。通过一个类的全限定名来获取其定义的二进制字节流(Class字节码),将这个字节流所代表的静态存储结构转化为方法区的运行时数据接口,根据字节码在java堆中生成一个代表这个类的java.lang.Class对象。
- 链接:
2.1.验证:验证Class文件中的字节流包含的信息是否符合当前虚拟机的要求。
2.2.准备:为静态域分配存储空间并设置类变量的初始值(默认值),
2.3.解析:将常量池中的符号引用转化为直接引用。 - 初始化:类的初始化顺序 :父类(静态变量、静态代码块)–>子类(静态变量、静态代码块)–>父类(变量、代码块)–> 父类构造器–>子类(变量、初始化块)–>子类构造器。注意:静态代码和静态变量同级,变量和代码块同级。谁在前先执行谁。类只会初始化一次。
补充:静态初始化,是在加载类的时候初始化。而非静态初始化,是new类实例对象的时候加载。
类什么时候才被初始化: 1.创建类的实例,2.访问类或接口的静态变量,调用类的静态方法 3.反射,4.初始化一个类的子类(会首先初始化子类的父类)
测试代码
public class Test {
public static void main(String[] args) {
try {
System.out.println("-------加载类-----");
System.out.println(Class.forName("Son"));
System.out.println("---new类对象实例------");
new Son();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
}
class Parent{
private static String a=inita();
private String b=initb();
static {
System.out.println("父类静态代码块开始执行");
}
{
System.out.println("父类代码块开始执行");
}
public static String inita(){
System.out.println("父类静态变量开始赋值");
return "父类静态变量";
}
public String initb(){
System.out.println("父类成员变量开始赋值");
return "父类成员变量";
}
public Parent(){
System.out.println("父类构造方式开始执行");
}
}
class Son extends Parent{
{
System.out.println("子类代码块开始执行");
}
static {
System.out.println("子类静态代码块开始执行");
}
private static String a=initSa();
private String sb=initSb();
public static String initSa(){
System.out.println("子类静态变量开始赋值");
return "子类静态变量";
}
public String initSb(){
System.out.println("子类成员变量开始赋值");
return "子类成员变量";
}
public Son(){
System.out.println("子类构造方式开始执行");
}
}
运行结果:-------加载类-----
父类静态变量开始赋值
父类静态代码块开始执行
子类静态代码块开始执行
子类静态变量开始赋值
class Son
—new类对象实例------
父类成员变量开始赋值
父类代码块开始执行
父类构造方式开始执行
子类代码块开始执行
子类成员变量开始赋值
子类构造方式开始执行