这票文章主要想来说一下Java类各部分(非静态字段初始化、非静态块、静态字段初始化、静态块、构造函数)的执行顺序。
1.不考虑继承的情况
写了一个demo来验证。构造了一个包含上述各个部分的类,观察在new一个这个类的对象时的输出。代码如下:
public class LoadOrderHelp {
static {
System.out.println("1:静态块");
b = 1;
//这一句报错(Illegal forward reference ),即不能在b没有声明初始化前输出,到可以赋值。咦!
//System.out.println(b);
}
public static int b = 10; //静态字段初始化
static {
System.out.println("2: 静态块 static b = " + b);
b = 5;
}
private int a = 1;//非静态字段初始化
{
System.out.println("3:非静态块 a = " + a + " b = " + b);
}
{
System.out.println("4:非静态块 a = " + a);
}
public LoadOrderHelp() {
System.out.println("5:构造函数 a = " + a +" b = " + b);
}
public static void main(String[] args) {
LoadOrderHelp loadOrderHelp = new LoadOrderHelp();
}
}
输出如下:
1:静态块
2: 静态块 static b = 10
3:非静态块 a = 1 b = 5
4:非静态块 a = 1
5:构造函数 a = 1 b = 5
从输出可以看出,总体上以静态内容->非静态内容->构造函数的顺序执行。
分析:
静态块和静态字段初始化是在类加载的时候就进行的,而非静态字段和块是在实例化对象时进行的,而构造函数又在字段和块之后执行,所以总体上呈现了上述顺序。
总结:
- 静态字段只在类第一次加载时初始化一次。
- 创建一个新对象new Dog()时,先在堆上为对象分配足够的内存,然后将这片内存清零(数字为0,布尔为false)引用为null。然后执行所有非静态字段的初始化(静态字段的初始化这时已经完成了),然后执行构造函数。
- 注意:java鼓励使用变量之前进行初始化,不应该依赖与默认的初始值。好的做法是总对变量进行初始化。
2. 考虑带父类的情况
有人总结为:父类静态变量——父类静态代码块——子类静态代码块——父类非静态变量——父类非静态代码块——父类构造函数——子类非静态变量——子类非静态代码块——子类构造函数
总体上呈先静态后非静态,先父类后子类,构造函数最后的顺序。