【Java】深入剖析Java枚举类

定义

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.可以方便的进行多态操作:枚举类支持方法的重写和多态,可以方便的扩展功能。

相关推荐
Re.不晚5 分钟前
Java入门15——抽象类
java·开发语言·学习·算法·intellij-idea
雷神乐乐11 分钟前
Maven学习——创建Maven的Java和Web工程,并运行在Tomcat上
java·maven
码农派大星。15 分钟前
Spring Boot 配置文件
java·spring boot·后端
顾北川_野22 分钟前
Android 手机设备的OEM-unlock解锁 和 adb push文件
android·java
江深竹静,一苇以航24 分钟前
springboot3项目整合Mybatis-plus启动项目报错:Invalid bean definition with name ‘xxxMapper‘
java·spring boot
confiself40 分钟前
大模型系列——LLAMA-O1 复刻代码解读
java·开发语言
Wlq04151 小时前
J2EE平台
java·java-ee
XiaoLeisj1 小时前
【JavaEE初阶 — 多线程】Thread类的方法&线程生命周期
java·开发语言·java-ee
杜杜的man1 小时前
【go从零单排】go中的结构体struct和method
开发语言·后端·golang
幼儿园老大*1 小时前
走进 Go 语言基础语法
开发语言·后端·学习·golang·go