如何为Java 6枚举实现values()?

les*_*es2 66 java enums

在Java中,您可以按如下方式创建枚举:

public enum Letter {
    A, B, C, D, E, F, G;

    static {
       for(Letter letter : values()) {
          // do something with letter
       }
    }
}
Run Code Online (Sandbox Code Playgroud)

这个问题涉及"values()"方法.具体来说,它是如何实现的?通常,我可以在Eclipse中使用F3或CTRL + Click跳转到Java类的源代码(即使对于String,Character,Integer甚至Enum这样的类).可以查看其他枚举方法的来源(例如,valueOf(String)).

"values()"每次调用时都会创建一个新数组吗?如果我将它分配给局部变量然后修改其中一个元素,会发生什么(显然这不会影响values()返回的值,这意味着每次都会分配一个新数组).

它的代码是原生的吗?或者JVM /编译器是否专门处理它,只有当它无法证明它不会被修改时才从values()返回一个新实例.

luc*_*smo 106

基本上,编译器(javac)将您的枚举转换为包含编译时所有值的静态数组.当您调用values()时,它会为您提供此数组的.clone'd()副本.

鉴于这个简单的枚举:

public enum Stuff {
   COW, POTATO, MOUSE;
}
Run Code Online (Sandbox Code Playgroud)

您实际上可以查看Java生成的代码:

public enum Stuff extends Enum<Stuff> {
    /*public static final*/ COW /* = new Stuff("COW", 0) */,
    /*public static final*/ POTATO /* = new Stuff("POTATO", 1) */,
    /*public static final*/ MOUSE /* = new Stuff("MOUSE", 2) */;
    /*synthetic*/ private static final Stuff[] $VALUES = new Stuff[]{Stuff.COW, Stuff.POTATO, Stuff.MOUSE};

    public static Stuff[] values() {
        return (Stuff[])$VALUES.clone();
    }

    public static Stuff valueOf(String name) {
        return (Stuff)Enum.valueOf(Stuff.class, name);
    }

    private Stuff(/*synthetic*/ String $enum$name, /*synthetic*/ int $enum$ordinal) {
        super($enum$name, $enum$ordinal);
    }
}
Run Code Online (Sandbox Code Playgroud)

您可以通过创建临时目录并运行来查看javac如何"翻译"您的类:

javac -d <output directory> -XD-printflat filename.java
Run Code Online (Sandbox Code Playgroud)

  • 为什么`values()`方法`clone()`数组`$ VALUES`?为什么不直接退货呢? (3认同)
  • @RestInPeace因为数组是可变的.有关示例,请参见http://stackoverflow.com/a/16125639/488861 (3认同)
  • 我不知道这个`-XD-printflat`。了解此类事物的内部运作非常有用。遗憾的是,这件事没有更好的记录。谢谢你让我意识到这一点! (2认同)