Enums - статические и экземплярные блоки
Я узнал, что в 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
статический блок
Ответы
Ответ 1
Вы должны знать, что элементы перечисления также являются статическими, а порядок вызова статического элемента зависит от его положения. См. Этот пример
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
Ответ 2
Немного поздно и создаем ответ Pshemo. Вывод кода (компиляции) ниже выглядит следующим образом:
8
10
Foo
static block
Bar
Итак, инициализация констант enum выполняется сначала (как сказано в 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"); } }
Ответ 3
1. Тип перечисления - это тип, поля которого состоят из фиксированного набора констант. Общие примеры включают направления компаса (значения NORTH, SOUTH, EAST и WEST) и дни недели.
2. Они статическая конечная константа, поэтому имеют все буквы в Caps.
3. И статические переменные инициализируются, как только JVM загружает класс.
Подробнее см. по этой ссылке:
http://docs.oracle.com/javase/1.5.0/docs/guide/language/enums.html
Ответ 4
Использовать байт-код, чтобы решить эту проблему.
import java.util.ArrayList;
import java.util.List;
public enum EnumDemo {
ONE(1), TWO(2);
private final static List<Integer> 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<EnumDemo> {
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 "<init>":(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 "<init>":(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."<init>":()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
}
Итак, экземпляр enum является статическим экземпляром и в начале.