java 枚举类_你需要关注的 Java Enum 枚举的几个细节

枚举是一个非常古老的语言特性,用来实现具名的有限集合,在 C/C++ 中使用广泛。而 Java 在 Java SE5 才引入枚举。也许语言设计者觉得既然是后引入该特性,那么一定要在这个特性上支持比其他语言更多的特性。这些特性的确让 Java 的枚举功能看起来更加“成熟”,同时也引入了一些复杂性,需要开发者关注。

枚举是一个不能继承的常规类

定义一个一周七天的枚举类型:

public enum EnumWeekDay {
    Mon, Tue, Wed, Thu, Fri, Sat, Sun;
}

编译成 class 文件后反编译查看:

╰─➤  javap EnumWeekDay
Compiled from "EnumWeekDay.java"
public final class EnumWeekDay extends java.lang.Enum<EnumWeekDay> {
  public static final EnumWeekDay Mon;
  public static final EnumWeekDay Tue;
  public static final EnumWeekDay Wed;
  public static final EnumWeekDay Thu;
  public static final EnumWeekDay Fri;
  public static final EnumWeekDay Sat;
  public static final EnumWeekDay Sun;
  public static EnumWeekDay[] values();
  public static EnumWeekDay valueOf(java.lang.String);
  static {};
}

从反编译结果可知:

  1. 枚举类型的关键字 enum 其实只是一个语法糖,编译器最终把它转化为一个final类,因此枚举是不可继承的。
  2. 枚举类型都继承自 java.lang.Enum 类。
  3. 枚举的每一个取值被编译器传化为了一个个 static final 属性。
  4. 本质上,这就是一个普通类,因此你可以在枚举是添加各种方法,甚至是main方法。

神奇的 values() 方法

从上面我们可以看出枚举类型被添加了一个静态的 values() 方法,但是 java.lang.Enum 并没有该方法。其实,这个方法是编译器添加的。通过这个方法可以获取到该枚举类型的所有取值。这个方法在需要遍历枚举取值,进行判断筛选的场景非常有用,可参考下例的 getByZhName 方法。

在枚举中保存其他信息

在 C 中,枚举可以简单的理解为具名的整型子集。Java 扩展了这个属性,使得可以在枚举中保存其他信息。

定义一个水果枚举类,并包含中文信息:

public enum EnumFruit {
    APPLE("苹果"),
    BANANA("香蕉"),
    ORANGE("橘子");

    private String zhName;

    EnumFruit(String zhName) {
        System.out.println("enum init:" + zhName);
        this.zhName = zhName;
    }

    /**
     * Getter method for property <tt>zhName</tt>.
     *
     * @return property value of zhName
     */
    public String getZhName() {
        return zhName;
    }

    public EnumFruit getByZhName(String zhName) {
        for (EnumFruit fruit : values()) {
            if (fruit.getZhName().equals(zhName)) {
                return fruit;
            }
        }

        return null;
    }
}

使用这种方式定义枚举的方式需要注意:该枚举必须含有一个构造函数,且该构造函数必须是私有的。因为枚举就是常规类,而枚举对象就是具体的枚举实例,因此枚举有多少个取值,该构造函数就会被调用多少次:

public class EnumUser {
    public static void main(String[] args) {
        EnumFruit fruit = EnumFruit.APPLE;
        System.out.println(fruit);
    }
}
enum init:苹果
enum init:香蕉
enum init:橘子
APPLE

使用 EnumSet 和 EnumMap 提供性能

如果要在把枚举使用在 Set、Map 等集合场景,请使用 EnumSet 和 EnumMap。 EnumSet 使用了 bit vector 来标记元素,EnumMap 内部将 Map 实现简化为了数组,因此可以获得更好的性能。

小结

Java 的枚举语言特性作为一个后来者,的确带来了更加“成熟”和“丰富”的实现。但是,这些丰富的特性是否一定要在日常的项目中使用,我个人是不推荐的。就我个人理解,枚举最大的优点是类型和有限集合的约束,从而增强代码的一致性。因此,我提倡在项目代码中用 C 的枚举风格来使用 Java 枚举。此外,枚举并不是编程语言必须支持的特性,比如近段时间如日中天的 Golang 是不支持枚举的。既然是一个可有可无的语言特性,那就 use is as simple as possible 吧。

扩展阅读

Java 语言中 Enum 类型的使用介绍 Java中的枚举与values()方法

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值