values - java enum со значениями




Почему мы разрешаем ссылаться на член enum члена enum в Java? (3)

Учитывая следующее перечисление:

enum Repeat {
    Daily,
    Weekly,
    Yearly
}

Я понимаю, что мы можем написать это так:

Repeat repeat = Repeat.Daily.Weekly.Yearly.Weekly;

что эквивалентно:

Repeat repeat = Repeat.Weekly;

Могу ли я знать, почему такой синтаксис разрешен? Есть ли способ, чтобы компилятор предупреждал нас об этом?


Есть ли способ, чтобы компилятор предупреждал нас об этом?

Да, используйте хорошую IDE и включите предупреждение. Таким образом, вы получите уведомление, как только напишите код, прежде чем даже скомпилировать его.

Например, в Eclipse это называется «Нестатический доступ к статическому члену»:


Литералы Enum являются статическими членами, и с каждым статическим членом можно получить к ним доступ, используя ссылку на класс:

TypeName.staticMember
TypeName.staticMethod()

Или на примере:

new TypeName().staticMember
new TypeName().staticMethod()

Второй подход не рекомендуется (и компилятор выдаст предупреждение)

Так как литералы enum являются просто статическими членами, Repeat.Daily.Weekly.Yearly.Weekly подобен второму фрагменту кода выше, получая доступ к статическим членам по ссылкам на экземпляры.

С классом это будет:

class Type {
    static Type INSTANCE1, INSTANCE2, INSTANCE3;
}

И можно получить ссылку на INSTANCE3 используя Type.INSTANCE1.INSTANCE2.INSTANCE3 . Это верно, но это плохая практика.


Это допустимо, так как Daily, Weekly, Yearly являются static полями по default внутри enum и содержат объект Repeat . Кроме того, вы получите предупреждение от компилятора "The static field Repeat.Weekly should be accessed in a static way" . Это похоже на строки кода ниже.

class Foo{
    public static Foo obj1 = new Foo();
    public static Foo obj2 = new Foo();
    public static Foo obj3 = new Foo();
}

Foo f = Foo.obj1.obj2.obj3; // will work fine but you will get a warning from the compiler.

Вот некоторая часть проверки байт-кода Repeat enum, и из этого ясно, что переменная Enum является static и содержит сам объект Enum.

   0: new           #1                  // class com/java8/demo/Repeat
   3: dup
   4: ldc           #14                 // String Daily
   6: iconst_0
   7: invokespecial #15                 // Method "<init>":(Ljava/lang/String;I)V
  10: putstatic     #19                 // Field Daily:Lcom/java8/demo/Repeat;
  13: new           #1                  // class com/java8/demo/Repeat 






enums