定义
1)定义
枚举类是Java 5引入的,在Java 5之前,Java并没有内置的枚举类型,只能通过自定义类来实现类似枚举的功能。例如:
java
public class EnumClass {
public static final int CONSTANT1 = xxx;
public static final int CONSTANT2 = xxx;
public static final int CONSTANT3 = xxx;
...
}
Java 枚举类是一种特殊类型的数据结构,一般用来存储定义一些字符串,数字等数据结构。枚举类中的每个常量都称为枚举常量。枚举类在Java中使用关键字enum定义。一般枚举类格式如下:
java
public enum EnumClass {
CONSTANT1,
CONSTANT2,
CONSTANT3,
...
}
其中,EnumClass为枚举类的名称,CONSTANT1、CONSTANT2、CONSTANT3等为枚举常量的名称。枚举类的引入,提供了一种更简洁、更安全的定义枚举类型的方式。
枚举类可以直接在其他类中引用,例如:
java
public enum Color {
RED,
GREEN,
BLUE
}
java
public class ColorUtil {
private ColorUtil() {
}
public static void main(String[] args) {
Color red = Color.RED;
}
}
枚举类是类,所以当然也可以在其他类内部定义:
java
public class Example {
public enum Color {
RED,
GREEN,
BLUE
}
public static void main(String[] args) {
System.out.println(Color.RED);
}
}
枚举类在内部定义时,可以将其作为内部类来定义,也可以将其作为静态内部类来定义,内部定义的枚举类可以访问外部类的成员变量和方法,比如:
java
public class Example {
private static int privateVariable = 10;
public enum Color {
RED, GREEN, BLUE;
public void print() {
System.out.println(name() + " has a value of " + Example.privateVariable);
}
}
public static void main(String[] args) {
Example.Color.RED.print();
}
}
这段代码运行main方法的结果为:
2)内部实现
我们借助IDEA中的一个反编译分析插件Jadx Class Decompiler反编译刚刚创建的Color枚举类,首先安装插件如下:
安装完成后,右键点击需要反编译的java文件,这里选择Color.java:
选择01 分析字节码,可以得到分析结果如下:
可以看到,枚举类是public和final修饰的,这表示它不能像普通的类一样被继承,也可以看到枚举类继承自java.lang.Enum类型,并且定义了public static final修饰的三个实例变量RED GREEN BLUE,这三个实例变量实际都是Color的实例对象。 此外其内部还定义了静态的values方法,它会返回一个包含所有枚举实例的Color[]结构的列表,以及valuesOf(String input)方法,它通过传入对应的枚举常量字符串,返回对应的Color实例。
3)方法与源码
我们翻阅官方文档可以得知java.lang.Enum包含了如下一些方法(查看Enum类源码也能看到):
其中特有方法应属valuesOf()和ordinal(),valuesOf()刚刚已经说明,它会返回指定名称的枚举常量,例如:
java
public enum Color {
RED,
GREEN,
BLUE;
}
Color color = Color.valueOf("GREEN");
ordinal()方法主要是获取枚举量在枚举类中的顺序,比如在上面例子中RED排在第一位,ordinal()返回值为0,示例如下:
java
public static void main(String[] args) {
Color red = Color.RED;
int ordinal = red.ordinal(); // ordinal = 0
}
另外在字节码分析结果中可以看到其实Enum是有构造方法的,但是这个构造方法用户无法调用,只有编译器可以调用,我们翻阅java.lang.Enum源码可以看到:
另外Enum还针对枚举常量实现了compareTo方法,并且这个compareTo方法默认是根据枚举的ordinal来对比的,也就是根据枚举在枚举类的声明顺序来对比的。
高级特性
1)switch用法
我们知道一般在使用switch进行判断时,可以使用整数、字符串,甚至表达式作为条件,但其实switch也支持枚举,并且不需要使用枚举的引用,代码示例如下:
java
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;
public class ColorUtil {
private ColorUtil() {
}
private static String getDescription(Color color) {
switch(color) {
case RED:
return "The color of blood";
case GREEN:
return "The color of grass";
case BLUE:
return "The color of the sky";
default:
return "";
}
}
public static void main(String[] args) {
Color red = Color.RED;
String description = getDescription(red);
System.out.println(description);
}
}
运行可以得到结果如下:
2)自定义传值与构造函数
这个特性我本人经常使用到,首先枚举类中可以定义好属性,然后自定义构造函数,在声明枚举类实例的时候传入对应的属性值,比如在业务中可能用到的isDelete字段,我们可以定义枚举类如下:
java
import lombok.Getter;
public enum IsDeleteEnum {
TRUE(1, "是"),
FALSE(0, "否"),
;
//值描述
@Getter
private String desc;
//枚举值
@Getter
private Integer code;
IsDeleteEnum(Integer code, String desc) {
this.code = code;
this.desc = desc;
}
}
这里包含了两个成员变量code和desc,分别代表枚举的值和描述信息。 这个类使用了Lombok库中的@Getter注解,枚举值和名称可以被Getter方法直接访问。
如果上面这个还是初级版本,那么下面这个高级版本则更为典型和常用,我们定义了ColorEnum的枚举类,它包含RED、BLUE、GREEN三个枚举值。每个枚举值都有一个值和名称,并且有一个静态的Map用于根据值获取枚举常量。这个类还提供了一个公共的静态方法of(),用于根据给定的值返回对应的枚举常量,如果没有找到则返回null。
java
import lombok.Getter;
import java.util.HashMap;
import java.util.Map;
public enum ColorEnum {
RED(0, "红色"),
BLUE(1, "蓝色"),
GREEN(2, "绿色"),
;
// 枚举值
@Getter
private Integer value;
@Getter
private String name;
private static Map<Integer, ColorEnum> map = new HashMap<>();
static {
for (ColorEnum r : ColorEnum.values()) {
map.put(r.getValue(), r);
}
}
ColorEnum(Integer value, String name) {
this.value = value;
this.name = name;
}
public static ColorEnum of(Integer value) {
return map.getOrDefault(value, null);
}
}
如果我想知道某个常量值x是否包含在枚举类ColorEnum 中,就可以使用of()方法:
java
public static void main(String[] args) {
System.out.println(ColorEnum.of(0) == null); // false
System.out.println(ColorEnum.of(3) == null); // true
}
运行结果如下:
3)枚举实现抽象方法
枚举中不只可以定义一些整数或是字符串常量,甚至可以重新实现抽象方法,我们定义了一个枚举类型Weekday,表示一周中的每一天。每个枚举常量都重写了抽象方法doSomething(),并实现了具体的业务逻辑,代码如下:
java
public enum Weekday {
MONDAY {
@Override
public void doSomething() {
System.out.println("Doing something on Monday");
}
},
TUESDAY {
@Override
public void doSomething() {
System.out.println("Doing something on Tuesday");
}
},
WEDNESDAY {
@Override
public void doSomething() {
System.out.println("Doing something on Wednesday");
}
},
THURSDAY {
@Override
public void doSomething() {
System.out.println("Doing something on Thursday");
}
},
FRIDAY {
@Override
public void doSomething() {
System.out.println("Doing something on Friday");
}
},
SATURDAY {
@Override
public void doSomething() {
System.out.println("Doing something on Saturday");
}
},
SUNDAY {
@Override
public void doSomething() {
System.out.println("Doing something on Sunday");
}
};
// 抽象方法
public abstract void doSomething();
public static void main(String[] args) {
Weekday day = Weekday.MONDAY;
day.doSomething();
}
}
在main函数中,通过Weekday的枚举常量来调用对应的方法,这样可以根据枚举常量来执行不同的操作,大大增加代码的可读性和可维护性。 这段代码执行结果为:
4)枚举注解属性
我们自定义一个名为ColorInterface的注解用于标记字段。注解定义一个value属性,类型为我们上文创建的的ColorEnum枚举类型。注解的保留策略为RetentionPolicy.RUNTIME,表示在运行时可以访问到该注解。
java
import java.lang.annotation.*;
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface ColorInterface {
ColorEnum value();
}
如下面代码所示,我们可以通过我们自定义的ColorInterface注解标记Flower类中的属性。main方法创建了一个Flower对象,使用反射获取Flower类中的Lily字段的注解,然后通过反射获取注解的value属性,也就是我们的枚举:
java
import java.lang.reflect.Field;
public class Flower {
@ColorInterface(ColorEnum.BLUE)
private int Lily;
@ColorInterface(ColorEnum.RED)
private int rose;
public static void main(String[] args) throws Exception {
Flower flower = new Flower();
Field field = Flower.class.getDeclaredField("Lily");
ColorInterface annotation = field.getAnnotation(ColorInterface.class);
ColorEnum value = annotation.value();
System.out.println(value);
}
}
代码运行如下:
5)枚举实现接口
Java枚举类还可以实现接口,可以为枚举类型添加更多的方法和行为,从而扩展枚举类的功能,枚举类可以具有更多的灵活性和可扩展性此外。
java
public interface Animal {
String makeSound();
String getName();
}
java
public enum Dog implements Animal {
BROWN("Buddy", "Buddy the Brown Dog"),
BLACK("Max", "Max the Black Dog")
;
private String name;
private String sound;
Dog(String name, String sound) {
this.name = name;
this.sound = sound;
}
@Override
public String makeSound() {
return sound;
}
@Override
public String getName() {
return name;
}
public static void main(String[] args) {
Animal dog = Dog.BLACK;
System.out.println(dog.getName()); // 输出: Max
System.out.println(dog.makeSound()); // 输出: Max the Black Dog
}
}
运行结果如下!:
总结
枚举类是Java中的原始类型之一,可以作为方法参数和返回值。枚举类可以通过类内的枚举常量表示特定的值。本文从源码和字节码实现的角度对枚举类的原理做了阐述,并且还展示了枚举类的多种高级用法,枚举类不仅可以像普通的类常量一样进行访问和使用,并且可以进行多态操作,不同的枚举常量可以有不同的方法实现,并且可以作为方法的参数和返回值。 因此枚举类有如下优点:
1.提高代码的可读性。枚举类的命名规范清晰明了,能够更直观地表达代码的含义,提高代码的可读性
2.更容易的调试和维护。枚举类中的方法和变量都在一个地方,方便调试和维护。
3.更好的类型安全性。使用枚举类可以提供更好的类型安全性,避免了使用整数或其他类型的错误。
4.可以方便的进行多态操作:枚举类支持方法的重写和多态,可以方便的扩展功能。