EffectiveJava(33) -- EnumMap详解(使用EnumMap代替序数)
2017-03-21 22:19
555 查看
因为数组不能与泛型兼容 , 所以当你要使用 ordinal 方法来索引数组的代码的时候,应该考虑使用 EnumMap .
EnumMap 在内部使用了数组,它也是Map的一个实现类 , Map 的实现类有很多种,EnumMap 从名字我们可以看出这个Map是给枚举类用的。它的key为枚举元素 ,value自定义。在工作中我们也可以用其他的Map来实现我们关于枚举的需求 ,但是为什么要用这个EnumMap呢?因为它的性能高 !为什么性能高?因为它的内部是用数组的数据结构来维护的!
例如 – 当你要按照类型将元素列出来 我们是用 enum 的 ordinal 和 EnumMap 分别实现进行比较
上面提到过 , 数组不能与泛型兼容 , 程序需要进行未受检转换 .因为数组不知道他的索引代表什么 , 你必须手工标注这些索引的输出 . 当你这么做时 , 访问一个按照枚举的序数进行索引的数组时 , 使用正确的 int 值就是你的指责了; 而在这个程序中使用 int 值是不安全的.
所以 , 我们可以使用 EnumMap 来快速实现这个功能
从上面的例子我们可以看到 , EnumMap内部使用了数组和 Map , 它集 Map 的丰富功能和类型安全的与数组的快速于一身 .
除此之外我们还可以用 EnumMap 对按照序数进行索引的数组的数组进行操作
这个例子模拟了自然界固态液态气态之间的转换 .
如果要添加一个 plasma 阶段 , 为了更新基于数组的程序 , 必须给 Phase 添加一种新常量 , 给 Phase.Transition 添加两种新常量 . 如果给数组添加元素过多或者过少或者放置元素不妥当 , 都会运行时失败 . 为此 , 我们必须将 plasma 添加到 Phase 列表 , 并将 IONIZE(GAS,PLASMA) 和 DEIONIZE(PLASMA,GAS) 添加到 Phase.Transition 的列表中 . 其他事情程序会自行处理 , 你几乎连出错机会都没有
总结: 最好不要用序数来索引数组 . 如果你所表示的这种关系是多维的 , 就使用 EnumMap<…,EnumMap<…>> . 应用程序的程序员在一般情况下都不使用 Enum.ordinal , 即使要用也是因为特殊情况 .
EnumMap 在内部使用了数组,它也是Map的一个实现类 , Map 的实现类有很多种,EnumMap 从名字我们可以看出这个Map是给枚举类用的。它的key为枚举元素 ,value自定义。在工作中我们也可以用其他的Map来实现我们关于枚举的需求 ,但是为什么要用这个EnumMap呢?因为它的性能高 !为什么性能高?因为它的内部是用数组的数据结构来维护的!
例如 – 当你要按照类型将元素列出来 我们是用 enum 的 ordinal 和 EnumMap 分别实现进行比较
public class Herb { public enum Type { ANNUAL, PERENNTAL, BIENNIAL } // private static final Herb Type = null; 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; } // 将集合放到一个按照类型的序数进行索引的数组中来实现 替换 Herb[] garden = { new Herb("1", Type.ANNUAL), new Herb("2", Type.BIENNIAL), new Herb("3", Type.PERENNTAL) }; Set<Herb>[] herbsByType = (Set<Herb>[])new Set[Herb.Type.values().length]; for(int i = 0;i<herbsByType.length;i++){ herbsByType[i] = new HashSet<Herb>(); } for(Herb h:garden){ herbsByType[h.type.ordinal()].add(h); } }}
上面提到过 , 数组不能与泛型兼容 , 程序需要进行未受检转换 .因为数组不知道他的索引代表什么 , 你必须手工标注这些索引的输出 . 当你这么做时 , 访问一个按照枚举的序数进行索引的数组时 , 使用正确的 int 值就是你的指责了; 而在这个程序中使用 int 值是不安全的.
所以 , 我们可以使用 EnumMap 来快速实现这个功能
Herb[] garden = { new Herb("1", Type.ANNUAL), new Herb("2", Type.BIENNIAL), new Herb("3", Type.PERENNTAL) }; Map<Herb.Type, Set<Herb>> herbsByType = new EnumMap<Herb.Type,Set<Herb>>(Herb.Type.class); for(Herb.Type t:Herb.Type.values()){ herbsByType.put(t, new HashSet<Herb>()); } for(Herb h:garden) herbsByType.get(h.type).add(h); System.out.println(herbsByType);
从上面的例子我们可以看到 , EnumMap内部使用了数组和 Map , 它集 Map 的丰富功能和类型安全的与数组的快速于一身 .
除此之外我们还可以用 EnumMap 对按照序数进行索引的数组的数组进行操作
这个例子模拟了自然界固态液态气态之间的转换 .
public enum Phase { SOILD,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; private Transition(Phase src,Phase dst) { this.src = src; this.dst = dst; } //由键为源Phase,值为map组成的Map,其中组成键的Map是由键值对目标Phase/Transition组成 private static final Map<Phase, Map<Phase,Transition>> m = new EnumMap<Phase, Map<Phase,Transition>>(Phase.class); static{ //初始化外部map,得到三个空的map for(Phase p:Phase.values()){ m.put(p, new EnumMap<Phase,Transition>(Phase.class)); } //利用每个状态过渡常量提供的起始信息和目标信息初始化内部的map for(Transition trans :Transition.values()){ m.get(trans.src).put(trans.dst, trans); } } public static Transition from(Phase src,Phase dst){ return m.get(src).get(dst); } } }
如果要添加一个 plasma 阶段 , 为了更新基于数组的程序 , 必须给 Phase 添加一种新常量 , 给 Phase.Transition 添加两种新常量 . 如果给数组添加元素过多或者过少或者放置元素不妥当 , 都会运行时失败 . 为此 , 我们必须将 plasma 添加到 Phase 列表 , 并将 IONIZE(GAS,PLASMA) 和 DEIONIZE(PLASMA,GAS) 添加到 Phase.Transition 的列表中 . 其他事情程序会自行处理 , 你几乎连出错机会都没有
总结: 最好不要用序数来索引数组 . 如果你所表示的这种关系是多维的 , 就使用 EnumMap<…,EnumMap<…>> . 应用程序的程序员在一般情况下都不使用 Enum.ordinal , 即使要用也是因为特殊情况 .
相关文章推荐
- Java中EnumMap代替序数索引代码详解
- 【Effective Java】9、使用EnumMap代替序数索引
- (33):用EnumMap代替序数索引
- Effective Java 1:考虑使用静态工厂方法代替构造器
- Java EnumMap 代替序数索引
- Effective Java 学习笔记 使用静态工厂方法代替构造器
- Effective Java(2nd Edition) Item 57 仅为例外条件使用异常(译文)
- Java开源技术:Eclipse的使用技巧详解
- 基于Java字符编码的使用详解
- 在Eclipse的Web项目中java里面使用ant进行数字签名步骤详解
- java odjc ResultSet 的使用详解
- Java下使用Oracle存储过程(详解)第1/3页
- HTTP头部详解及使用Java套接字处理HTTP请求
- Eclipse3使用JSF来设计页面详解(java)
- java clone方法使用详解
- 使用Java操作文本文件的方法详解
- HTTP头部详解及使用Java套接字处理HTTP请求
- Java开源技术:Eclipse的使用技巧详解
- 使用Java操作文本文件的方法详解
- tcl/tk实例详解——修改目录下所有文件(使用一个字符串代替另外一个)