什么是枚举

枚举是JDK 1.5中引入的新特性,由一组固定的常量组成合法值的类型,例如一年中的季节、一周的星期数。枚举其实就是特殊的类,继承了java.lang.Enum类,并实现了java.lang.Seriablizablejava.lang.Comparable两个接口。域成员均为常量,且构造方法被默认为私有。

如何定义枚举

先来看看枚举是如何定义的!我们定义四个值,分别为 春天、夏天、秋天、冬天

public enum SeasonEnum {
    //春天 
    SPRING,
    //夏天
    SUMMER,
    //秋天
    AUTUMN,
    //冬天
    WINTER;
}
复制代码

以上方式就是枚举类的定义方式。很简单!那我们再来看看枚举类提供的函数。

枚举的函数

public static void main(String[] args) {
    //1、根据传入的名称获得指定的枚举,可能会抛出异常
    SeasonEnum autumn = SeasonEnum.valueOf("AUTUMN");
    System.out.println("autumn1 = " + autumn);
    //与 1、一致
    SeasonEnum anEnum = SeasonEnum.valueOf(SeasonEnum.class, "AUTUMN");
    System.out.println("autumn2 = " +anEnum);
    //2、返回当前枚举类中的所有元素
    SeasonEnum[] values = SeasonEnum.values();
    System.out.println("values = " +Arrays.toString(values));
    //3、获得枚举元素
    SeasonEnum name1 = SeasonEnum.AUTUMN;
    System.out.println("name1 = " + name1);
    //获得枚举元素的名称
    String name2 = SeasonEnum.AUTUMN.name();
    System.out.println("name2 = " + name2);
    //4、返回此枚举元素的索引,从 0 开始
    int ordinal = SeasonEnum.AUTUMN.ordinal();
    System.out.println("ordinal = " + ordinal);
    //5、与指定的对象进行比较,返回一个负整数,零或正整数。
    //小于指定对象返回 负整数
    //等于指定对象返回 零
    //大于指定对象 返回正整数
    int i = SeasonEnum.AUTUMN.compareTo(SeasonEnum.AUTUMN);
    System.out.println("compareTo = " + i);
    //返回枚举类的类型
    System.out.println("getDeclaringClass = " + SeasonEnum.AUTUMN.getDeclaringClass());
    //如果指定的对象等于此枚举元素,则返回true。
    System.out.println("equals = " + SeasonEnum.AUTUMN.equals("AUTUMN"));

}

结果:
autumn1 = AUTUMN
autumn2 = AUTUMN
values = [SPRING, SUMMER, AUTUMN, WINTER]
name1 = AUTUMN
name2 = AUTUMN
ordinal = 2
compareTo = 0
getDeclaringClass = class com.gongj.jsondate.controller.SeasonEnum
equals = false
复制代码

枚举的使用

上面已经简单的介绍了枚举的定义枚举的函数!那本节就带大家来看看在工作当中如何去使用枚举,哪些地方可以去使用枚举!

1、定义常量

就用上述的SeasonEnum枚举类。

public static void main(String[] args) {
    SeasonEnum type = SeasonEnum.AUTUMN;
    if(SPRING.equals(type)){
        System.out.println("春天:春暖花开");
    }
    if(SUMMER.equals(type)){
        System.out.println("夏天:夏阳酷暑");
    }
    if(AUTUMN.equals(type)){
        System.out.println("秋天:秋风习习");
    }
    if(WINTER.equals(type)){
        System.out.println("冬天:冬日暖阳");
    }
}
结果:秋天:秋风习习
复制代码

ifswitch里的判断语句值都能用枚举进行替代,提高代码可读性。

2、参数接收

接口的请求参数值可以用枚举进行接收!比如OrderDTO类的orderType字段的类型,就可以使用枚举进行接收!那有什么好处呢?

  • 1、代码可读性,会让其他开发者,一眼就知道订单类型有哪一些类型(值。
  • 2、明确订单类型的范围。可以防止用户随意传值。
public class OrderDTO {

   private Long id;

   private String orderName;

   private SeasonEnum orderType;
}
复制代码

这里还是用SeasonEnum枚举类来演示。

提供对外接口

@PostMapping("/save")
public void save(@RequestBody OrderDTO req){
    System.out.println( JSON.toJSONString(req));
}
复制代码

然后进行调用:http://localhost:8080/save,响应提示 value 不是声明的 Enum 实例名称之一

image.png

也就是说orderType的值,只能为 SeasonEnum枚举类所声明的实例。

3、码值转换

使用枚举类实现可以省略掉许多的 if/else。大多数用于对接不同的系统,比如:接到一个与银行对接的功能,流程如下:前端 -》 本系统后端 -》调用银行接口。 其中有一个支付状态的码值。在自己系统 1-待支付,而在银行那边 0-待支付。两个系统之间的码值不一致,所以本系统就需要配置转换规则。而这时候就可以使用枚举类来进行实现。

3.1、编写枚举基类

编写枚举基类,所有枚举类都需要实现该接口,如果基类满足不了需求,子类可以随意扩展。

public interface BaseEnum {

    String getKey();

    void setKey(String key);

    String getValue();

    void setValue(String value);

    String getDesc();

    void setDesc(String desc);
}
复制代码

3.2、编写支付枚举类

实现BaseEnum接口,并自行扩展 channelchannelDesc两个字段,并增加match方法。

public enum PayEnum implements BaseEnum{
    WAITING_PAY("1","0","待支付","unionpay","银联"),
    SUCCESS_PAY("2","1","支付成功","unionpay","银联"),
    FAIL_PAY("3","2","支付失败","unionpay","银联"),

    ALIPAY_WAITING_PAY("1","3","待支付","alipay","支付宝");

    private String key;
    private String value;
    private String desc;

    //本类扩展字段 用于对接不同系统的支付状态
    private String channel;
    private String channelDesc;

    private PayEnum(String key,String value,String desc,String channel,String channelDesc){
        this.key = key;
        this.value = value;
        this.desc = desc;
        this.channel = channel;
        this.channelDesc = channelDesc;
    }
    @Override
    public String getKey() {
        return this.key;
    }

    @Override
    public void setKey(String key) {
       return;
    }

    @Override
    public String getValue() {
        return this.value;
    }

    @Override
    public void setValue(String value) {
       return;
    }

    @Override
    public String getDesc() {
        return this.desc;
    }

    @Override
    public void setDesc(String desc) {
        return;
    }

    public String getChannel() {
        return channel;
    }

    public void setChannel(String channel) {
        return;
    }

    public String getChannelDesc() {
        return channelDesc;
    }

    public void setChannelDesc(String channelDesc) {
         return;
    }

    /**
     * 根据 key 和 channel 获得枚举实例
     * @param key
     * @param channel
     * @return
     */
    public static PayEnum match(String key, String channel) {
        PayEnum[] enums = PayEnum.values();
        for (PayEnum payEnum : enums) {
            if (key.equals(payEnum.getKey()) && channel.equals(payEnum.getChannel()))
                return payEnum;
        }
        return null;
    }
}
复制代码

3.3、测试

public static void main(String[] args) {
    String key = "1";
    String channel = "unionpay";
    PayEnum match = match(key, channel);
    System.out.println(match.getKey() + "==" + match.getValue() 
    + "===" + match.getDesc() + "===" + 
    "==" + match.getChannel());
}
结果:1==0===待支付=====unionpay
复制代码

在某些情况使用枚举,可以省略掉非常多的if/else