有时候,你可能会见到利用ordinal方法(第35项)索引到数组或列表的代码。例如用下面这个简单的类来表示植物:
class Plant {
enum LifeCycle {
ANNUAL, PERENNIAL, BIENNIAL }
final String name;
final LifeCycle lifeCycle;
Plant(String name, LifeCycle lifeCycle) {
this.name = name;
this.lifeCycle = lifeCycle;
}
@Override public String toString() {
return name;
}
}
现在假设有一个Plant数组来表示一座花园的植物,你想要根据它们的生命周期(一年生、多年生或者两年生)进行组织之后将这些植物列出来。如果要这么做的话,需要构建三个集合,每个生命周期一个集合,并遍历整座花园,将每种植物放到相应的集合中。一些程序猿会通过将这些集合放到一个按照生命周期的序数进行索引的数组中来实现这一点。
// Using ordinal() to index into an array - DON'T DO THIS!
Set<Plant>[] plantsByLifeCycle = (Set<Plant>[]) new Set[Plant.LifeCycle.values().length];
for (int i = 0; i < plantsByLifeCycle.length; i++)
plantsByLifeCycle[i] = new HashSet<>();
for (Plant p : garden)
plantsByLifeCycle[p.lifeCycle.ordinal()].add(p);
// Print the results
for (int i = 0; i < plantsByLifeCycle.length; i++) {
System.out.printf("%s: %s%n", Plant.LifeCycle.values()[i], plantsByLifeCycle[i]);
}
这种方法的确可行,但是隐藏着许多问题。因为数组不能与泛型兼容(第28项),程序需要进行未受检的转换,并且不能正确无误地进行编译。因为数组不知道它的索引代表着什么,你必须手工标注(label)这些索引的输出。但是这种方法最引种的问题在于,当你访问一个按照枚举的序数进行索引的数组时,使用正确的int值就是你的职责了,int不能提供枚举的类型安全。你如果使用了错误的值,程序就会悄悄地完成错误的工作,如果幸运的话,会抛出ArrayIndexOutOfBoundException异常。
这有一种更好的办法可以达到同样的效果。数组实际上充当着从枚举到值的映射,因此可能还要用到Map。更具体地说,有一种非常快速的Map实现专门用于枚举键,称作java.util.EnumMap。以下就是用EnumMap改写后的程序:
// Using an EnumMap to associate data with an enum
Map<Plant.LifeCycle, Set