枚举 – 静态和实例块

我已经了解到,在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; } } 

输出:

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

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

 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(); } } 

产量

 static block 1 creating SomeClass object static block 2 

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

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

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

 8 10 Foo static block Bar 

因此首先执行枚举常量初始化(如Pshemo所说,它们始终是隐式staticfinal ,请参见第二个blockquote)然后初始化显式声明为static所有字段。 如前所述,语言规范说明了类初始化期间的执行顺序以及枚举常量:

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

除了枚举类型Einheritance自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"); } } 

枚举类型是一种类型,其字段由一组固定的常量组成 。 常见示例包括罗盘方向(NORTH,SOUTH,EAST和WEST的值)和星期几。

它们是静态最终常量 ,因此在Caps中有所有字母。

3.一旦JVM加载类,就会初始化静态变量。

有关详细信息,请参阅此链接:

http://docs.oracle.com/javase/1.5.0/docs/guide/language/enums.html

使用字节码来解决这个问题。

 import java.util.ArrayList; import java.util.List; public enum EnumDemo { ONE(1), TWO(2); private final static List vals; static { System.out.println("fetch instance from static"); vals = new ArrayList<>(); EnumDemo[] values = EnumDemo.values(); for (EnumDemo value : values) { vals.add(value.val); } } private int val; EnumDemo(int val){ this.val = val; System.out.println("create instance:" + val); } } 

使用javac编译到类文件,然后javap -c EnumDemo.class ,得到了这个:

 Compiled from "EnumDemo.java" public final class EnumDemo extends java.lang.Enum { public static final EnumDemo ONE; public static final EnumDemo TWO; public static EnumDemo[] values(); Code: 0: getstatic #1 // Field $VALUES:[LEnumDemo; 3: invokevirtual #2 // Method "[LEnumDemo;".clone:()Ljava/lang/Object; 6: checkcast #3 // class "[LEnumDemo;" 9: areturn public static EnumDemo valueOf(java.lang.String); Code: 0: ldc_w #4 // class EnumDemo 3: aload_0 4: invokestatic #5 // Method java/lang/Enum.valueOf:(Ljava/lang/Class;Ljava/lang/String;)Ljava/lang/Enum; 7: checkcast #4 // class EnumDemo 10: areturn static {}; Code: 0: new #4 // class EnumDemo 3: dup 4: ldc #16 // String ONE 6: iconst_0 7: iconst_1 8: invokespecial #17 // Method "":(Ljava/lang/String;II)V 11: putstatic #18 // Field ONE:LEnumDemo; 14: new #4 // class EnumDemo 17: dup 18: ldc #19 // String TWO 20: iconst_1 21: iconst_2 22: invokespecial #17 // Method "":(Ljava/lang/String;II)V 25: putstatic #20 // Field TWO:LEnumDemo; 28: iconst_2 29: anewarray #4 // class EnumDemo 32: dup 33: iconst_0 34: getstatic #18 // Field ONE:LEnumDemo; 37: aastore 38: dup 39: iconst_1 40: getstatic #20 // Field TWO:LEnumDemo; 43: aastore 44: putstatic #1 // Field $VALUES:[LEnumDemo; 47: getstatic #8 // Field java/lang/System.out:Ljava/io/PrintStream; 50: ldc #21 // String fetch instance from static 52: invokevirtual #15 // Method java/io/PrintStream.println:(Ljava/lang/String;)V 55: new #22 // class java/util/ArrayList 58: dup 59: invokespecial #23 // Method java/util/ArrayList."":()V 62: putstatic #24 // Field vals:Ljava/util/List; 65: invokestatic #25 // Method values:()[LEnumDemo; 68: astore_0 69: aload_0 70: astore_1 71: aload_1 72: arraylength 73: istore_2 74: iconst_0 75: istore_3 76: iload_3 77: iload_2 78: if_icmpge 109 81: aload_1 82: iload_3 83: aaload 84: astore 4 86: getstatic #24 // Field vals:Ljava/util/List; 89: aload 4 91: getfield #7 // Field val:I 94: invokestatic #26 // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer; 97: invokeinterface #27, 2 // InterfaceMethod java/util/List.add:(Ljava/lang/Object;)Z 102: pop 103: iinc 3, 1 106: goto 76 109: return } 

因此,枚举实例是静态实例,并且在头部。