当枚举在类中时,在枚举构造函数内创建类的新实例

问题描述 投票:1回答:1

最初我以为我会在这里遇到鸡蛋和鸡蛋问题,但是在单元测试中探索这个并不表示有任何问题。我想了解这里发生了什么。我原以为,因为枚举是静态的,所以当JVM加载MyEnum时,MyClass构造函数会运行。但是在我的测试中,它在“MyEnum构造函数”之前打印“getValue”。

MyClass {

   private enum MyEnum {
       VALUE;

       MyEnum() {
          System.out.println("MyEnum constructor");
          MyClass clazz = new MyClass(); 
       }
   }

   public MyEnum getValue() {
      System.out.println("getValue");
      return MyEnum.VALUE;
   }

}

public class MyClassTest {

    @Test
    public void testStuff() {
        MyClass clazz = new MyClass();
        clazz.getValue();
    }
}
java enums
1个回答
1
投票

类/接口/枚举/注释嵌套在另一个类中的事实不会影响其初始化何时发生。

无论如何都适用初始化规则。它们在JLS here中定义。

类或接口类型T将在第一次出现以下任何一个之前立即初始化:

  • T是一个类,并且创建了T的实例。
  • 调用由T声明的静态方法。
  • 分配由T声明的静态字段。
  • 使用由T声明的静态字段,该字段不是常量变量(第4.12.4节)。

关于says类型,JLS还有enum

枚举声明指定一个新的枚举类型,一种特殊的类类型。

它说,还有about its members

对于在E声明体中声明的每个枚举常量c,E有一个隐含声明的public static final字段,其类型为E,与c同名。该字段有一个变量初始值设定项,它实例化E并将c的任何参数传递给为E选择的构造函数。该字段与c(如果有)具有相同的注释。

把所有这些放在一起,你会得到你所看到的行为的解释。

你的代码实例化MyClass,然后调用它的getValue()方法。 getValue()打印出标准输出,然后尝试访问MyEnum声明的静态字段。这会触发枚举类型的初始化,初始化public static static VALUE字段,该字段调用相应的MyEnum构造,再次打印到标准输出。

© www.soinside.com 2019 - 2024. All rights reserved.