枚举 - 静态和实例块

NIN*_*OOP 33 java enums static-block

我已经了解到,在Java中,静态块在初始化类时执行,并且在构造每个类的实例之前执行实例块.我总是在实例块之前看到要执行的静态块.为什么案件相反enums

任何人都可以请解释示例代码的输出:

enum CoffeeSize {

    BIG(8), LARGE(10),HUGE(12),OVERWHELMING();
    private int ounces ;

    static {
        System.out.println("static block ");
    }
    {
        System.out.println("instance block");
    }

    private CoffeeSize(int ounces){
        this.ounces = ounces;
        System.out.println(ounces);
    }
    private CoffeeSize(){
        this.ounces = 20;
        System.out.println(ounces);
    }

    public int getOunces() {
        return ounces;
    }
} 
Run Code Online (Sandbox Code Playgroud)

输出:

实例块
8
实例块
10
实例块
12
实例块
20
静态块

Psh*_*emo 34

您需要知道枚举值是包含该枚举类型实例的静态字段,静态字段的初始化顺序取决于它们的位置.看这个例子

class SomeClass{
    public SomeClass() { System.out.println("creating SomeClass object"); }
}

class StaticTest{
    static{ System.out.println("static block 1"); }
    static SomeClass sc = new SomeClass();
    static{ System.out.println("static block 2"); }

    public static void main(String[] args) {
        new StaticTest();
    }
}
Run Code Online (Sandbox Code Playgroud)

产量

static block 1
creating SomeClass object
static block 2
Run Code Online (Sandbox Code Playgroud)

现在因为枚举值始终位于枚举类型的开头,所以它们将始终在任何静态初始化块之前被调用,因为其他所有内容都只能在枚举值之后声明.
但是初始化枚举值(在类初始化时发生)它们的构造函数被调用,正如你所说的那样,非静态初始化块在每个构造函数的开头执行,这就是你看到它们的原因:

  • 对于每个枚举值
  • 在任何静态初始化块之前.

  • 这就是为什么您应该将它们全部写成大写,这表示常量是“ static final”,也就是常量。 (2认同)
  • @Pshemo:但是我无法在该枚举中定义常量之前定义静态块,这会导致编译错误。这是否意味着对于所有枚举,将首先初始化静态最终常量,然后再执行其他任何静态块? (2认同)

Wol*_*ram 9

很晚才建立在Pshemo的答案上.下面(编译)代码的输出如下:

8
10
Foo
static block 
Bar
Run Code Online (Sandbox Code Playgroud)

因此首先执行枚举常量初始化(如Pshemo所说,它们总是隐式地,static并且final参见第二个块引用)然后显式声明的所有字段static都被初始化.如上所述,语言规范说明了类初始化期间的执行顺序以及枚举常量:

接下来,按文本顺序执行类的类变量初始值设定项和类的静态初始值设定项,或接口的字段初始值设定项,就好像它们是单个块一样.

除了枚举类型E从Enum继承的成员之外,对于每个声明的名称为n的枚举常量,枚举类型都有一个隐式声明的名为n的类型为E的公共静态最终字段.在枚举类型中显式声明的任何静态字段之前,这些字段被认为与相应的枚举常量以相同的顺序声明.


class StaticTest {
    enum CoffeeSize {
        BIG(8), LARGE(10);
        private int ounces;

        static Foo foo = new Foo();
        static { System.out.println("static block "); }
        static Bar bar = new Bar();

        private CoffeeSize(int ounces){
            this.ounces = ounces;
            System.out.println(ounces);
        }
    }
    public static void main(String[] args) {
        CoffeeSize cs = CoffeeSize.LARGE;
    }
}

class Foo { public Foo() { System.out.println("Foo"); } }
class Bar { public Bar() { System.out.println("Bar"); } }
Run Code Online (Sandbox Code Playgroud)