A question a day ,get offer every day.
-- 未来的魔法师
类加载过程分为:加载,(验证,准备,解析)连接,初始化,使用,卸载。
准备阶段:
在准备阶段会对静态变量先进行赋值,且置为初始值,在本例中为null。如果静态变量如:
public final static String p_StaticField = "父类--静态变量";
那么在准备阶段就将直接赋值为"父类--静态变量",否则将留到初始化阶段进行赋值操作。
初始化阶段:
初始化阶段主要是执行<clinit>的过程,<clinit>()方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块(static{}块)中的 语句合并产生的。如果没有静态变量和者静态语句块,那么它就不会执行。同时Java虚拟机会保证在子类的<clinit>()方法执行前,父类的<clinit>()方法已经执行完毕。
执行new关键字会调用<init>()方法,其方法在<clinit>()执行完成后。与<clinit>()一样其首先需要对父类的初始化。
得到以下结论:
1. 父类中的内容先执行
2. 静态变量、静态代码块先执行且只执行一遍
3. new 关键字调用一次,实例化对象一次。
下面以一段代码阐述父子类的初始化顺序情况。定义例子中对象内容如下:
父类(Parent)包括:静态变量,静态块,变量,构造器
子类(Son)包括:静态变量,静态块,变量,构造器
/**
* @Author jlu
* @Date 11:08 2022/4/12
* @Description:
**/
//父类
public class Parent {
// 静态变量
public static String p_StaticField = "父类--静态变量";
// 变量(其实这用对象更好能体同这一点,如专门写一个类的实例)
//如果这个变量放在初始化块的后面,是会报错的,因为你根本没有被初始化
public String p_Field = "父类--变量";
// 静态初始化块
static {
System.out.println(p_StaticField);
System.out.println("父类--静态初始化块");
}
// 初始化块
{
System.out.println(p_Field);
System.out.println("父类--初始化块");
}
// 构造器
public Parent() {
System.out.println("父类--构造器");
}
}
//子类
public class Son extends Parent {
// 静态变量
public static String s_StaticField = "子类--静态变量";
// 变量
public String s_Field = "子类--变量";
// 静态初始化块
static {
System.out.println(s_StaticField);
System.out.println("子类--静态初始化块");
}
// 初始化块
{
System.out.println(s_Field);
System.out.println("子类--初始化块");
}
// 构造器
public Son() {
//super();
System.out.println("子类--构造器");
}
}
public class Test {
public static void main(String[] args) {
System.out.println("*************first load***************");
new Son();
System.out.println("*************second sonClass***************");
new Son();
}
}
输出结果:
*************first load***************
//初始化
父类--静态变量
父类--静态初始化块
子类--静态变量
子类--静态初始化块
//实例化
父类--变量
父类--初始化块
父类--构造器
子类--变量
子类--初始化块
子类--构造器
*************second sonClass***************
父类--变量
父类--初始化块
父类--构造器
子类--变量
子类--初始化块
子类--构造器