最初我以为我会在这里遇到鸡蛋和鸡蛋问题,但是在单元测试中探索这个并不表示有任何问题。我想了解这里发生了什么。我原以为,因为枚举是静态的,所以当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();
}
}
类/接口/枚举/注释嵌套在另一个类中的事实不会影响其初始化何时发生。
无论如何都适用初始化规则。它们在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
构造,再次打印到标准输出。