
在Java编程中,当我们需要将枚举类型作为键来存储数据时,EnumMap是比HashMap更优的选择。EnumMap是专门为枚举键设计的Map实现,它在内部使用数组存储值,因此具有极高的性能,并且是类型安全的。在处理涉及枚举状态转换等复杂映射场景时,EnumMap的优势尤为突出。
考虑一个典型的场景:定义物质的不同相(固态、液态、气态)及其相互之间的转换。我们可以用一个外部枚举Phase表示物质的相,用一个嵌套枚举Transition表示相之间的转换。每个Transition实例都包含一个源相(from)和一个目标相(to)。我们的目标是构建一个映射,能够通过源相和目标相快速查找对应的Transition实例。例如,从液态到固态的转换是“凝固”(FREEZE)。
在Java的早期版本,或者在不倾向于使用Stream API的场景下,初始化复杂的EnumMap通常采用显式的循环结构。这种方法通常涉及两层循环:外层循环用于初始化每个枚举键对应的内部EnumMap,内层循环则遍历所有转换实例,将其放入对应的映射中。
以下是《Effective Java》第二版中可能采用的初始化方式:
// Using a nested EnumMap to associate data with enum pairs
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);
final Phase src; // 源相
final Phase dst; // 目标相
Transition(Phase src, Phase dst) {
this.src = src;
this.dst = dst;
}
// 初始化相转换映射
private static final Map<Phase, Map<Phase, Transition>> m =
new EnumMap<Phase, Map<Phase, Transition>>(Phase.class);
static {
// 第一步:为每个源相初始化一个内部的EnumMap
for (Phase p : Phase.values()) {
m.put(p, new EnumMap<Phase, Transition>(Phase.class));
}
// 第二步:遍历所有转换实例,将其放入对应的内部EnumMap中
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);
}
}
}这种方法的优点是逻辑清晰、直观易懂。通过分步操作,我们可以清楚地看到Map是如何被构建和填充的。对于Java初学者或习惯命令式编程风格的开发者来说,这种方式的可读性较高。然而,其缺点是代码相对冗长,尤其是在映射关系更加复杂时,可能需要更多的循环和条件判断。
随着Java 8引入Stream API,集合操作变得更加声明式和函数式。对于EnumMap的初始化,尤其是需要根据多个属性进行分组和映射的场景,Stream API提供了一种更为简洁高效的解决方案。
以下是《Effective Java》第三版中采用的Stream API初始化方式:
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 from; // 源相
private final Phase to; // 目标相
Transition(Phase from, Phase to) {
this.from = from;
this.to = to;
}
// 初始化相转换映射
private static final Map<Phase, Map<Phase, Transition>>
m = Stream.of(values()).collect(groupingBy(t -> t.from, // 按源相分组
toMap(t -> t.to, t -> t, // 内部Map:键为目标相,值为转换实例
(x, y) -> y, // 合并函数:在键冲突时选择第二个值(此处不会发生冲突,但必须提供)
() -> new EnumMap<>(Phase.class)))); // 内部Map的工厂:确保生成EnumMap
public static Transition from(Phase from, Phase to) {
return m.get(from).get(to);
}
}
}这段代码利用了Stream.of(values())将所有Transition实例转换为一个流。接着,collect()方法结合了两个重要的收集器:
这种Stream API的初始化方式显著减少了代码行数,代码更加紧凑和声明式。它表达了“我们想要根据源相分组,然后在每个组内,根据目标相映射到转换实例”的意图,而不是“先创建这个,再遍历那个来填充”。
无论是采用传统的显式循环还是现代的Stream API,核心思想都是为了高效地初始化EnumMap,以实现枚举对之间的复杂映射。在实际项目中,选择哪种初始化方法应基于以下考量:
总之,EnumMap是处理枚举键映射的强大工具,而Java语言的发展为我们提供了多种初始化其复杂结构的方式。理解并灵活运用这些方法,将有助于我们编写出更高效、更可维护的Java代码。
以上就是EnumMap 初始化策略演进:从显式循环到 Stream API的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号