如何用Java实现枚举?

时间:2011-11-07 02:09:31

标签: java enums

我现在才意识到他们使用Constants类的力量和实用性......但我很有兴趣知道它们是如何实现的。它们似乎很像静态方法或静态常量,因为您只需导入“拥有”枚举的文件,您可以使用enumName.valueName来引用它们。 Javadocs在线似乎暗示他们是class但是根据需要提供“无名”课程似乎很奇怪。 (至少在Java中......)

3 个答案:

答案 0 :(得分:17)

我相信枚举的每个实例都是枚举的匿名最终子类。

反编译:

public enum Color {
    r(0xff0000), g(0x00ff00), b(0x0000ff);
    Color(int rgb) {
        this.rgb=rgb;
    }
    final int rgb;
}

您可以看到正在制作的实例:

D:\home\ray\dev\indigoapps\so8032067enumimpl\bin>javap -c Color
Compiled from "Color.java"
public final class Color extends java.lang.Enum{
public static final Color r;

public static final Color g;

public static final Color b;

final int rgb;

static {};
  Code:
   0:   new     #1; //class Color
   3:   dup
   4:   ldc     #16; //String r
   6:   iconst_0
   7:   ldc     #17; //int 16711680
   9:   invokespecial   #18; //Method "<init>":(Ljava/lang/String;II)V
   12:  putstatic       #22; //Field r:LColor;
   15:  new     #1; //class Color
   18:  dup
   19:  ldc     #24; //String g
   21:  iconst_1
   22:  ldc     #25; //int 65280
   24:  invokespecial   #18; //Method "<init>":(Ljava/lang/String;II)V
   27:  putstatic       #26; //Field g:LColor;
   30:  new     #1; //class Color
   33:  dup
   34:  ldc     #28; //String b
   36:  iconst_2
   37:  sipush  255
   40:  invokespecial   #18; //Method "<init>":(Ljava/lang/String;II)V
   43:  putstatic       #29; //Field b:LColor;
   46:  iconst_3
   47:  anewarray       #1; //class Color
   50:  dup
   51:  iconst_0
   52:  getstatic       #22; //Field r:LColor;
   55:  aastore
   56:  dup
   57:  iconst_1
   58:  getstatic       #26; //Field g:LColor;
   61:  aastore
   62:  dup
   63:  iconst_2
   64:  getstatic       #29; //Field b:LColor;
   67:  aastore
   68:  putstatic       #31; //Field ENUM$VALUES:[LColor;
   71:  return

public static Color[] values();
  Code:
   0:   getstatic       #31; //Field ENUM$VALUES:[LColor;
   3:   dup
   4:   astore_0
   5:   iconst_0
   6:   aload_0
   7:   arraylength
   8:   dup
   9:   istore_1
   10:  anewarray       #1; //class Color
   13:  dup
   14:  astore_2
   15:  iconst_0
   16:  iload_1
   17:  invokestatic    #43; //Method java/lang/System.arraycopy:(Ljava/lang/Obj
ect;ILjava/lang/Object;II)V
   20:  aload_2
   21:  areturn

public static Color valueOf(java.lang.String);
  Code:
   0:   ldc     #1; //class Color
   2:   aload_0
   3:   invokestatic    #51; //Method java/lang/Enum.valueOf:(Ljava/lang/Class;L
java/lang/String;)Ljava/lang/Enum;
   6:   checkcast       #1; //class Color
   9:   areturn

}

答案 1 :(得分:1)

如果您对如何实施它们感兴趣而不是对它们可以做什么感兴趣,只需查看their definition

编辑4年:Grr,不可靠的4岁链接...我还没有找到当前链接的良好链接,但您可以随时查找jdk来源或只使用wayback machine

答案 2 :(得分:0)

枚举是在首次加载枚举类时静态创建的,并且是不可变的。

如果要为枚举指定不同的值,则必须在枚举类中有一个构造函数。构造函数完成后,您无法更改枚举值(如上所述不可变)。

您不需要在枚举上使用equals()方法:==运算符可以正常工作。

我在我的博客中写了一个教程,展示了枚举类的一些不错的用法,这些只是阅读枚举引用并不简单。

如果你有兴趣,那就是链接。 Blog