Java枚举必须用enum关键字声明,是继承java.lang.Enum的final类,不可继承或new实例;枚举常量须在最前,自动为public static final实例;可含私有构造方法、字段和方法,带参常量需在无参之前。

Java 枚举必须用 enum 关键字声明,不能用 class 或 interface
Java 枚举是独立的引用类型,语法上必须以 enum 开头,后面紧跟枚举名和花括号。它不是类的变体,也不是接口实现——编译器会自动将其编译为继承 java.lang.Enum 的 final 类,因此你无法手动继承它,也不能用 new 实例化。
常见错误是试图写成 public class Status extends Enum,这会直接编译失败;或者漏掉花括号里的枚举常量,比如只写 enum Color { } 而没列值,虽然语法合法但毫无意义。
-
enum声明体中,枚举常量必须出现在最前面(可带参数或大括号定义内部类) - 每个枚举常量默认是
public static final的该枚举类型实例 - 枚举类可以有构造方法、字段、普通方法,但构造方法必须是
private(显式或隐式)
枚举常量后跟括号表示调用带参构造方法
当你看到 RED(255, 0, 0) 这样的写法,说明该枚举定义了接受三个 int 参数的私有构造方法。括号里的值会传给构造方法,用于初始化每个常量的实例字段。
不带括号的常量(如 GREEN)会走无参构造;混合写法也允许,比如 RED(255, 0, 0), GREEN, BLUE(0, 0, 255) —— 注意逗号分隔,且所有带参常量必须在无参之前(否则编译报错 Enum constant must be the first statement)。
立即学习“Java免费学习笔记(深入)”;
enum Color {
RED(255, 0, 0),
GREEN(0, 255, 0),
BLUE(0, 0, 255);
private final int r, g, b;
Color(int r, int g, int b) {
this.r = r;
this.g = g;
this.b = b;
}
public int getRGB() {
return (r << 16) | (g << 8) | b;
}
}
不要在 switch 中用 String 代替枚举常量
Java 7+ 支持 switch(String),但如果你已有枚举,就别绕路转成字符串再 switch。这样做不仅丢失类型安全,还会让 case 分支失去编译期检查:新增枚举值时,IDE 不会提示你补全 switch 分支,容易遗漏逻辑。
正确做法是直接 switch 枚举变量本身。JVM 对枚举 switch 有优化(底层转为 tableswitch/lookupswitch),性能不输 if-else,且支持 default 和编译检查。
- 用
switch(color) { case RED: ... case GREEN: ... },而不是switch(color.name()) - 如果真要匹配字符串,优先考虑用枚举的静态工厂方法,比如
Color.fromName("RED"),而非在 switch 里硬编码字符串 - 注意:
name()返回声明时的字面量(如"RED"),而toString()可被重写,不保证一致
序列化时用 name() 而非 toString() 或字段
Java 默认序列化枚举只保存其 name(),反序列化时靠这个名字查回原始常量。这是 JVM 级别的保证,稳定可靠。如果你重写了 toString(),或依赖某个业务字段(如 code 或 desc)做 JSON 序列化,就可能在跨服务或版本升级时出问题。
例如用 Jackson 时,默认行为是序列化为 name 字符串;若改成 @JsonValue 指向某个字段,那前端拿到的就不是标准枚举标识,后续加新值或改字段值都可能破坏兼容性。
- 对外暴露的 API 枚举字段,建议保持
name()语义清晰(如PENDING、PAID) - 需要展示文本时,用单独的方法(如
getLabel())返回,不要覆盖toString() - 数据库存枚举,推荐存
name()或自增序号(用ordinal()要小心——删/调换常量顺序会导致数据错乱)










