Java类的初始化顺序

这票文章主要想来说一下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. 考虑带父类的情况

有人总结为:父类静态变量——父类静态代码块——子类静态代码块——父类非静态变量——父类非静态代码块——父类构造函数——子类非静态变量——子类非静态代码块——子类构造函数
总体上呈先静态后非静态,先父类后子类,构造函数最后的顺序。

你可能感兴趣的:(Java类的初始化顺序)