之前提过,我们可能会用ordinal()方法来求索引,代替一些数组的索引,例如数种的例子,表示一些植物的归类
public class Herb {
public enum Type { ANNUAL, PERENNIAL, BIENNIAL }
private final String name;
private final Type type;
Herb(String name, Type type) {
this.name = name;
this.type = type;
}
@Override
public String toString() {
return name;
}
}
假设植物是按照一年 两年 多年 植物进行罗列,那么需要三个集合,每个集合装自己对应的植物,把植物园中所有的植物便利一遍,按照要求放进去,书中代码
// 我添加的,假设有四种植物,分别是 aa bb cc dd ,对应的的是枚举的三个类型值,分别代表 一年 两年 多年,最终打印的 // 是 一年植物的集合 二年植物的集合 多年植物的集合
Herb[] garden = {new Herb("aa", Herb.Type.ANNUAL), new Herb("bb", Herb.Type.PERENNIAL), new Herb
("cc", Herb.Type.BIENNIAL), new Herb("dd", Herb.Type.ANNUAL)};
Set
for(int i=0; i < herbsByType.length; i++)
herbsByType[i] = new HashSet
for(Herb h : garden)
herbsByType[h.type.ordinal()].add(h);
for(int i=0; i < herbsByType.length; i++)
System.out.printf("%s: %s%n", Herb.Type.values()[i], herbsByType[i]);
上述代码没问题,但有个可能出错的地方就是泛型与数组公用,我们知道,编译时泛型生效,运行时擦除泛型,如果类型错误的话,会报错的,这里比较好的是,我们没用错,那就解读一下这几行代码吧。我们创建了个Set集合类型的数组,就是说herbsByType[]数组中的每个对象都是个Set集合,并且数组中元素的个数是Herb.Type.values().length,值为3;然后便利这个数组,往这个数组里添加Set集合对象,此时 herbsByType[i] = new HashSet
ANNUAL: [dd, aa]
PERENNIAL: [bb]
BIENNIAL: [cc]
上述分析了一堆,看起来也比较麻烦,书中给出了另外一种写法。之前介绍了EnumSet,属于Set集合;今天介绍EnumMap,当然属于Map集合。
Herb[] garden = {new Herb("aa", Herb.Type.ANNUAL), new Herb("bb", Herb.Type.PERENNIAL), new Herb
("cc", Herb.Type.BIENNIAL),new Herb("dd", Herb.Type.ANNUAL)};
Map
for(Herb.Type t : Herb.Type.values())
herbByType.put(t, new HashSet
for(Herb h : garden)
herbByType.get(h.type).add(h);
System.out.println(herbByType);
这种写法,更安全,不必要泛型手动输入索引计算等等,也没有不安全转换,意思和上面的写法一样,只是在这里,用了Map集合代替了之前的数组,上面代码是数组中嵌套了Set集合,数组用索引来区分植物的年限;这里是用Map集合,key值为植物年限,value值为Set集合,除此之外,其他的都一样,打印结果为
{ANNUAL=[dd, aa], PERENNIAL=[bb], BIENNIAL=[cc]}
EnumMap 是属于Map集合,class EnumMap
如果我们碰到两次使用序数索引的数组,即二维数组,怎么变,看下面例子
public enum Phase {
SOLID, LIQUID, GAS;
public enum Transition {
MELT, FREEZE, BOTL, CONDENSE, SUBLIME, DEPOSIT;
private static final Transition[][] TRANSITIONS = {
{null, MELT, SUBLIME},
{FREEZE, null, BOTL},
{DEPOSIT, CONDENSE, null}
};
public static Transition from(Phase src, Phase dst) {
return TRANSITIONS[src.ordinal()][dst.ordinal()];
}
}
}
这段代码的意思是说,要进行二维数组组合,通过方法去取对应的值,看起来没问题,但实际上如果和第一个案例差不多,并且如果添加想新的植物,扩展不太方便,同理,可以使用EnumMap来修改,
public enum Phase {
SOLID, LIQUID, GAS;
public enum Transition {
MELT(SOLID,LIQUID), FREEZE(LIQUID, SOLID),
BOIL(LIQUID, GAS), CONDENSE(GAS, LIQUID),
SUBLIME(SOLID, GAS), DEPOSIT(GAS, SOLID);
private final Phase src;
private final Phase dst;
Transition(Phase src, Phase dst) {
this.src = src;
this.dst = dst;
}
private static final Map
new EnumMap
static {
for(Phase p : Phase.values())
m.put(p, new EnumMap
for(Transition t : Transition.values())
m.get(t.src).put(t.dst, t);
}
public static Transition from(Phase src, Phase dst) {
return m.get(src).get(dst);
}
}
}
这种修改方法,有点类似第二十九条中的泛型案例,把可能出现的情况,全都列举了出来,然后把他们装进一个map集合,然后map集合嵌套集合,这个好处就是避免了数组,不用担心类型异常和数据越界等问题。在枚举中,也要记得,优先使用EnumMap和EnumSet,避免使用数组。