一尘不染

Java静态初始化顺序

java

我正在尝试发现初始化发生的顺序,或者更确切地说,为什么要按此顺序进行初始化的原因。给定代码:

public class Main {

    {
        System.out.printf("NON-STATIC BLOCK\n");
    }

    static{
        System.out.printf("STATIC BLOCK\n");
    }

    public static Main m = new Main();

    public Main(){
        System.out.printf("MAIN CONSTRUCTOR\n");
    }

    public static void main(String... args) {
        //Main m = new Main();
        System.out.printf("MAIN METHOD\n");

    }
}

输出:

STATIC BLOCK

NON-STATIC BLOCK

MAIN CONSTRUCTOR

MAIN METHOD

但是,将m的声明移动到初始化块之前会产生:

NON-STATIC BLOCK

MAIN CONSTRUCTOR

STATIC BLOCK

MAIN METHOD

而且我完全不知道为什么会以这种顺序发生。此外,如果我static在的声明中消除了关键字m,则init块和构造函数均不会触发。谁能帮我这个忙吗?


阅读 175

收藏
2020-09-09

共1个答案

一尘不染

我认为您只是缺少JLS的12.4.2节,其中包括:

接下来,以文本顺序执行类的类变量初始化器和静态初始化器,或者接口的字段初始化器,就好像它们是单个块一样。

“按文本顺序”部分是重要的部分。

如果您m从静态变量更改为实例变量,则该字段将不会通过 初始化进行初始化-它只会通过 实例
初始化进行初始化(即,在构造实例时)。目前,这将导致堆栈溢出-创建一个实例需要创建另一个实例,这需要创建另一个实例,依此类推。

编辑:同样,第12.5节指定实例初始化,包括以下步骤:

* 执行此类的实例初始值设定项和实例变量初始值设定项,并按从左到右的顺序将实例变量初始值设定项的值分配给相应的实例变量,这些变量在文本中显示在该类的源代码中。如果执行这些初始化程序中的任何一个导致异常,则不会再处理其他初始化程序,并且该过程会因相同的异常而突然完成。否则,请继续执行步骤5。

  • 执行此构造函数的其余部分。如果该执行突然完成,则出于相同原因,此过程也会突然完成。否则,此过程将正常完成。

这就是为什么在“主构造器”之前看到“非静态块”的原因。

2020-09-09