Java中enum本质是继承自java.lang.Enum的final类,每个常量为静态不可变实例,支持字段、方法、构造器、接口实现及多态行为,天然线程安全单例,values()返回副本数组,valueOf()为O(n)查找。

Enum不只是常量集合,它本质是类
Java中的enum不是语法糖,而是在编译期生成的**继承自java.lang.Enum的final类**。每个枚举常量都是该类的一个静态不可变实例。这意味着枚举天然支持方法定义、字段、构造器,甚至可实现接口——它完全具备普通类的能力,只是语法更简洁、语义更严谨。
可以定义私有字段和构造器,封装状态
枚举实例可以携带数据。比如表示HTTP状态码时,不仅需要名称,还需要状态码数字和描述:
public enum HttpStatus {
OK(200, "Success"),
NOT_FOUND(404, "Resource not found"),
SERVER_ERROR(500, "Internal server error");
private final int code;
private final String reason;
HttpStatus(int code, String reason) {
this.code = code;
this.reason = reason;
}
public int getCode() { return code; }
public String getReason() { return reason; }
}
每个枚举常量在类加载时通过构造器初始化,并持有独立的状态,安全且不可变。
立即学习“Java免费学习笔记(深入)”;
支持重写方法与多态行为
枚举常量可以各自实现抽象方法,实现“每个实例不同行为”的效果(类似策略模式):
public enum Operation {
ADD {
@Override public int apply(int a, int b) { return a + b; }
},
MULTIPLY {
@Override public int apply(int a, int b) { return a * b; }
};
public abstract int apply(int a, int b);
}
这种写法比用switch判断更面向对象,也更易扩展,IDE还能自动补全所有分支,避免遗漏。
内置可靠单例与线程安全保障
每个枚举常量都是JVM级别唯一的实例,由类加载机制保证初始化顺序和线程安全性。无需双重检查锁或static final holder,就能获得最简、最安全的单例:
public enum ConfigLoader {
INSTANCE;
private Properties props;
ConfigLoader() {
props = loadFromDisk(); // 构造器中初始化
}
public Properties getConfig() { return props; }
}
它天然防止反射攻击(Enum构造器被JVM特殊保护)、序列化绕过,比手写单例更健壮。
values()和valueOf()背后是编译器生成的静态数组与查找逻辑
每次调用MyEnum.values(),实际返回的是编译器自动生成的private static final MyEnum[] $VALUES副本——所以它是不可修改的数组拷贝,线程安全但有内存开销;valueOf(String)则通过简单遍历该数组匹配name()字段,时间复杂度O(n),不适合高频调用。如需高性能查找,建议用Map缓存:
private static final Map
.collect(Collectors.toMap(HttpStatus::name, Function.identity()));
注意:不要用ordinal()做业务逻辑依据,它依赖声明顺序,极易因增删枚举项而悄然出错。










