EnumMap是专为枚举设计的高效映射,基于数组存储,键必须为枚举类型,具有O(1)查找性能、有序遍历和类型安全特性,适用于状态机、策略分发等场景。

在Java中,EnumMap 是专为枚举类型设计的高性能映射实现。它内部基于数组存储,保证了快速的访问速度和有序性。与普通HashMap不同,EnumMap要求键必须是枚举类型,这使得它在处理枚举键映射时更加高效、安全且易于维护。
EnumMap的基本结构特点
EnumMap是java.util包中的类,继承自AbstractMap,并实现了Map接口。其核心特性包括:
- 键类型限定为枚举:构造时需指定枚举类类型,不允许null键。
- 内部使用数组存储:根据枚举常量的ordinal值作为索引,实现O(1)级别的查找性能。
- 元素有序:遍历时按枚举声明顺序输出,具有天然的顺序性。
- 非线程安全:需要外部同步控制多线程访问。
由于这些设计,EnumMap非常适合用于将枚举常量映射到特定值或行为的场景,比如状态机、配置映射、策略分发等。
如何创建和使用EnumMap
使用EnumMap的第一步是定义一个枚举类型。例如:
立即学习“Java免费学习笔记(深入)”;
public enum Status {PENDING, APPROVED, REJECTED, CANCELLED
}
// 创建EnumMap并填充数据
EnumMap
statusMessages.put(Status.PENDING, "等待审核");
statusMessages.put(Status.APPROVED, "已通过");
statusMessages.put(Status.REJECTED, "已拒绝");
statusMessages.put(Status.CANCELLED, "已取消");
// 获取对应信息
String msg = statusMessages.get(Status.APPROVED); // 输出:已通过
注意构造函数传入的是枚举类的Class对象,这是EnumMap识别键类型的方式。泛型确保类型安全,避免错误类型插入。
典型应用场景示例
EnumMap常见于需要将枚举与行为或数据绑定的场合:
示例:用EnumMap实现简单的策略模式
public enum Operation {ADD, SUBTRACT, MULTIPLY, DIVIDE;
private static final EnumMap
new EnumMap(Operation.class);
// 静态初始化
static {
OPERATORS.put(ADD, (a, b) -> a + b);
OPERATORS.put(SUBTRACT, (a, b) -> a - b);
OPERATORS.put(MULTIPLY, (a, b) -> a * b);
OPERATORS.put(DIVIDE, (a, b) -> a / b);
}
public int apply(int a, int b) {
if (b == 0 && this == DIVIDE) throw new ArithmeticException("除零异常");
return OPERATORS.get(this).apply(a, b);
}
}
调用方式简洁直观:Operation.ADD.apply(5, 3) 返回8。
性能与注意事项
EnumMap相比HashMap在枚举键场景下有明显优势:
- 内存更紧凑,无哈希冲突开销。
- 迭代更快,顺序固定且连续。
- 类型安全强,编译期检查键类型。
但也要注意:
- 不能使用null作为键。
- 只适用于枚举类型,灵活性低于HashMap。
- 所有枚举常量都会预留位置(即使未放入映射),因此大枚举可能浪费空间。
基本上就这些。合理使用EnumMap能让代码更清晰、高效,特别是在枚举驱动的逻辑中表现尤为出色。










