1.在没有枚举之前,我们如果需要一些常量,比如说,我们想用一些常量来代替订单的几种状态,如已下单未付款、已付款未发货、已发货未确认收货、已收货未评价、已评价。我们会定义一个用来装常量的类,比如:



package com.xdx.learn;

public class OrderConstant {
    public static final int UNPAY=1;//未付款
    public static final int UNDELIVER=2;//未发货
    public static final int UNRECEIVE=3;//未收货
    public static final int UNCOMMENT=4;//未评价

}



OrderConstant .UNPAY就可以获取到这个常量。

  2.有了枚举类型以后,我们会这样来写代码。

  新建一个枚举类。

  



public enum OrderEnum {
    UNPAY("unpay",1),UNDELIVER("undeliver",2),UNRECEIVE("unreceive",3),UNCOMMENT("uncomment",4);
    private String key;
    private int value;
    private OrderEnum(String key,int value){
        this.key=key;
        this.value=value;
    }
    
    public String getKey() {
        return key;
    }

    public void setKey(String key) {
        this.key = key;
    }

    public int getValue() {
        return value;
    }

    public void setValue(int value) {
        this.value = value;
    }

    public static void main(String args[]){
        System.out.println(OrderEnum.UNPAY.getKey());
        System.out.println(OrderEnum.UNPAY.getValue());
    }

}



 

  上面就是一个枚举类,它有如下特点。

  (1)它不用class修饰,而是用enum关键字来修饰。但是要知道的是,它本质上还是一个类。

  (2)它的构造函数不能用public修饰,只能用private来修饰,也就是说,我们不能在外部实例化一个枚举类的对象。这让你想到了什么呢?是不是单例模式。

UNPAY("unpay",1),UNDELIVER("undeliver",2),UNRECEIVE("unreceive",3),UNCOMMENT("uncomment",4);这几个都是该枚举类的对象(他们都是OrderEnum类型的),以静态常量的成员变量的形式存在于枚举类中。事实上,他们是public static final类型的,所以我们可以在类外部使用类名.成员变量,比如OrderEnum.UNPAY的形式来访问。

  (4)一旦你定义了一个枚举类,则必须也将它的实例创建出来,即是上述的UNPAY("unpay",1)这些实例。实例的创建被简化了,只需要调用构造函数,不需要用new关键字。

  其实,按照我的理解,上述的枚举类可以用以下的类来代替。

  



package com.xdx.learn;

public class OrderMulti {
    private String key;
    private int value;
    private OrderMulti(String key,int value){
        this.key=key;
        this.value=value;
    }
    
    public String getKey() {
        return key;
    }
    public void setKey(String key) {
        this.key = key;
    }
    public int getValue() {
        return value;
    }
    public void setValue(int value) {
        this.value = value;
    }

    public static final OrderMulti UNPAY=new OrderMulti("unpay", 1);
    public static final OrderMulti UNDELIVER=new OrderMulti("undeliver", 2);
    public static final OrderMulti UNRECEIVE=new OrderMulti("unreceive", 3);
    public static final OrderMulti UNCOMMENT=new OrderMulti("uncomment", 4);
    public static void main(String args[]){
        System.out.println(OrderMulti.UNPAY.getKey());
        System.out.println(OrderMulti.UNPAY.getValue());
    }

}



  没错,枚举类就相当于一个带有多例(多例模式)的java类。只不过java的语法帮我们做了这些显式实例化的操作,并且以一种比较简单的语法来表示。就变成了enum了。

  3.再深入一点,其实枚举类都是Enum类的子类,去查jdk源码,发现Enum是一个抽象的泛型类,其定义为public abstract class Enum<E extends Enum<E>>
        implements Comparable<E>, Serializable。事实上,上述的OrderEnum类,可以理解成这样的一个类。

  public class OrderEnum extends Enum<OrderEnum>,没错,泛型的类型实参就是这个枚举类本身。

  不过当你真的在eclipse里面敲入上面的一个类,会发现报错,因为Enum这个类是不可继承的,提示的错误是The type OrderEnum may not subclass Enum<A> explicitly。不能显式的继承Enum。jdk在编译阶段就拒绝了一个类去继承Enum,具体什么原因,怎么实现,我也不知道。我们只需要知道enum修饰的类,它的父类是Enum就行了。

  既然如此,enum修饰的类也就不可以在继承其他的类了,因为java是单继承的。当然可以通过实现接口的方式去对enum类进行扩展。

  由于枚举类继承自Enum,那么Enum里面的一些方法他也可以用。看如下代码,使用了几个比较常用的方法。

  



public static void main(String args[]){
        System.out.println(OrderEnum.UNPAY.getKey());
        System.out.println(OrderEnum.UNPAY.getValue());
        //name()方法获取该枚举类实例的名称
        System.out.println(OrderEnum.UNPAY.name());
        //ordinal()方法获取该枚举类实例在所有实例中的排序,从0开始。
        System.out.println(OrderEnum.UNPAY.ordinal());
        //compareTo()方法比较两个枚举实例的排序,可认为是前者的ordinal-后者的ordinal的值。
        System.out.println(OrderEnum.UNPAY.compareTo(OrderEnum.UNDELIVER));
        System.out.println(OrderEnum.UNRECEIVE.compareTo(OrderEnum.UNDELIVER));
        System.out.println(OrderEnum.UNCOMMENT.compareTo(OrderEnum.UNDELIVER));
        //获取该枚举对象的类
        System.out.println(OrderEnum.UNPAY.getDeclaringClass());
        //验证枚举类的父类确实是Enum
        System.out.println(OrderEnum.UNPAY.getDeclaringClass().getSuperclass());
        System.out.println(OrderEnum.UNPAY.equals(OrderEnum.UNCOMMENT));
        //遍历枚举类中实例
        for(OrderEnum orderenum:OrderEnum.values()){
            System.out.println(orderenum.getKey());
        }
    }



  上述代码的运行结果为:

  unpay
  1
  UNPAY
  0
  -1
  1
  2
  class com.xdx.learn.OrderEnum
  class java.lang.Enum
  false
  unpay
  undeliver

  unreceive
  uncomment

  4.只要将枚举理解成一个实现了多例模式的类,运用起来就不会有什么困难。但是也有人会问,我用第一种方式,即直接使用一个public static final int UNPAY=1。这样的常量。不是也可以实现枚举需要的功能吗?为何还大费周章去定义一个枚举类呢?我觉得是基于如下几方面考虑的。

  (1)首先枚举类的类名可以有一定的指示作用,比如我们给一个枚举类命名为week,我们可以知道它应该就是代表星期,而在枚举之前,我们使用常量的容器类,往往只定义一个类,命名为类似于Constant这样的类,要去里面找寻其中的常量值是比较费劲的。

  (2)当用枚举作为函数的形参的时候,能起到限定的作用。比如我有一个函数 ,我可以定义为void func(int x),接受一些常量值。我也可以定义成void func(OrderEnum orderEnum)这样的形式。后者比前者的优点在于它限制了传入的参数只能是该枚举类的实例,而前者则可以传入任意整型。