Java中的Enum的使用与分析

来源:互联网 发布:js 所见即所得编辑器 编辑:程序博客网 时间:2024/06/15 18:41

JDK1.5引入了新的类型——枚举。

用法一:常量

JDK1.5 之前,我们定义常量都是: publicstaticfianl.... 。现在好了,有了枚举,可以把相关的常量分组到一个枚举类型里,而且枚举提供了比常量更多的方法。

 

Java代码 
  1. public enum Color {  
  2.   RED, GREEN, BLANK, YELLOW  
  3. }  

 

用法二:switch

JDK1.6之前的switch语句只支持int,char,enum类型,使用枚举,能让我们的代码可读性更强。

 

Java代码 
  1. enum Signal {  
  2.     GREEN, YELLOW, RED  
  3. }  
  4. public class TrafficLight {  
  5.     Signal color = Signal.RED;  
  6.     public void change() {  
  7.         switch (color) {  
  8.         case RED:  
  9.             color = Signal.GREEN;  
  10.             break;  
  11.         case YELLOW:  
  12.             color = Signal.RED;  
  13.             break;  
  14.         case GREEN:  
  15.             color = Signal.YELLOW;  
  16.             break;  
  17.         }  
  18.     }  
  19. }  

 

用法三:向枚举中添加新方法

如果打算自定义自己的方法,那么必须在enum实例序列的最后添加一个分号。而且 Java 要求必须先定义 enum 实例。

 

Java代码 
  1. public enum Color {  
  2.     RED("红色"1), GREEN("绿色"2), BLANK("白色"3), YELLO("黄色"4);  
  3.     // 成员变量  
  4.     private String name;  
  5.     private int index;  
  6.     // 构造方法  
  7.     private Color(String name, int index) {  
  8.         this.name = name;  
  9.         this.index = index;  
  10.     }  
  11.     // 普通方法  
  12.     public static String getName(int index) {  
  13.         for (Color c : Color.values()) {  
  14.             if (c.getIndex() == index) {  
  15.                 return c.name;  
  16.             }  
  17.         }  
  18.         return null;  
  19.     }  
  20.     // get set 方法  
  21.     public String getName() {  
  22.         return name;  
  23.     }  
  24.     public void setName(String name) {  
  25.         this.name = name;  
  26.     }  
  27.     public int getIndex() {  
  28.         return index;  
  29.     }  
  30.     public void setIndex(int index) {  
  31.         this.index = index;  
  32.     }  
  33. }  

 

用法四:覆盖枚举的方法

下面给出一个toString()方法覆盖的例子。

 

Java代码 
  1. public enum Color {  
  2.     RED("红色"1), GREEN("绿色"2), BLANK("白色"3), YELLO("黄色"4);  
  3.     // 成员变量  
  4.     private String name;  
  5.     private int index;  
  6.     // 构造方法  
  7.     private Color(String name, int index) {  
  8.         this.name = name;  
  9.         this.index = index;  
  10.     }  
  11.     //覆盖方法  
  12.     @Override  
  13.     public String toString() {  
  14.         return this.index+"_"+this.name;  
  15.     }  
  16. }  

 

用法五:实现接口

所有的枚举都继承自java.lang.Enum类。由于Java 不支持多继承,所以枚举对象不能再继承其他类。

 

Java代码 
  1. public interface Behaviour {  
  2.     void print();  
  3.     String getInfo();  
  4. }  
  5. public enum Color implements Behaviour{  
  6.     RED("红色"1), GREEN("绿色"2), BLANK("白色"3), YELLO("黄色"4);  
  7.     // 成员变量  
  8.     private String name;  
  9.     private int index;  
  10.     // 构造方法  
  11.     private Color(String name, int index) {  
  12.         this.name = name;  
  13.         this.index = index;  
  14.     }  
  15. //接口方法  
  16.     @Override  
  17.     public String getInfo() {  
  18.         return this.name;  
  19.     }  
  20.     //接口方法  
  21.     @Override  
  22.     public void print() {  
  23.         System.out.println(this.index+":"+this.name);  
  24.     }  
  25. }  

 

用法六:使用接口组织枚举

 

Java代码 
  1. public interface Food {  
  2.     enum Coffee implements Food{  
  3.         BLACK_COFFEE,DECAF_COFFEE,LATTE,CAPPUCCINO  
  4.     }  
  5.     enum Dessert implements Food{  
  6.         FRUIT, CAKE, GELATO  
  7.     }  
  8. }  


  9. 用法七:关于枚举集合的使用

    java.util.EnumSet和java.util.EnumMap是两个枚举集合。EnumSet保证集合中的元素不重复;EnumMap中的 key是enum类型,而value则可以是任意类型。关于这个两个集合的使用就不在这里赘述,可以参考JDK文档

    枚举和常量定义的区别

    一、 通常定义常量方法

    我们通常利用public final static方法定义的代码如下,分别用1表示红灯,3表示绿灯,2表示黄灯。

    ?
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    publicclassLight {
        /* 红灯 */
        public final static int RED = 1;
     
        /* 绿灯 */
        public final static int GREEN = 3;
     
        /* 黄灯 */
        publicfinalstatic int YELLOW = 2;
      }

    二、 枚举类型定义常量方法

    枚举类型的简单定义方法如下,我们似乎没办法定义每个枚举类型的值。比如我们定义红灯、绿灯和黄灯的代码可能如下:

    ?
    1
    2
    3
    publicenumLight {
        RED, GREEN, YELLOW;
      }

    我们只能够表示出红灯、绿灯和黄灯,但是具体的值我们没办法表示出来。别急,既然枚举类型提供了构造函数,我们可以通过构造函数和覆写toString方法来实现。首先给Light枚举类型增加构造方法,然后每个枚举类型的值通过构造函数传入对应的参数,同时覆写toString方法,在该方法中返回从构造函数中传入的参数,改造后的代码如下:

    ?
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    publicenumLight {
     
      // 利用构造函数传参
      RED(1), GREEN(3), YELLOW(2);
     
      // 定义私有变量
      privateintnCode;
     
      // 构造函数,枚举类型只能为私有
      privateLight(int_nCode) {
     
        this.nCode = _nCode;
     
      }
     
      @Override
      publicString toString() {
     
        returnString.valueOf(this.nCode);
     
      }
     
    }

    三、 完整示例代码

    枚举类型的完整演示代码如下:

    ?
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    96
    97
    98
    99
    100
    101
    102
    103
    104
    105
    106
    107
    108
    109
    110
    111
    112
    113
    114
    115
    116
    117
    118
    119
    120
    121
    122
    123
    124
    125
    126
    127
    128
    129
    130
    publicclassLightTest {
     
      // 1.定义枚举类型
     
      publicenumLight {
     
        // 利用构造函数传参
     
        RED(1), GREEN(3), YELLOW(2);
     
        // 定义私有变量
     
        privateintnCode;
     
        // 构造函数,枚举类型只能为私有
     
        privateLight(int_nCode) {
     
          this.nCode = _nCode;
     
        }
     
        @Override
        publicString toString() {
     
          returnString.valueOf(this.nCode);
     
        }
     
      }
     
      /**
       *
       * @param args
       */
     
      publicstaticvoid main(String[] args) {
     
        // 1.遍历枚举类型
     
        System.out.println("演示枚举类型的遍历 ......");
     
        testTraversalEnum();
     
        // 2.演示EnumMap对象的使用
     
        System.out.println("演示EnmuMap对象的使用和遍历.....");
     
        testEnumMap();
     
        // 3.演示EnmuSet的使用
     
        System.out.println("演示EnmuSet对象的使用和遍历.....");
     
        testEnumSet();
     
      }
     
      /**
       *
       * 演示枚举类型的遍历
       */
     
      privatestaticvoid testTraversalEnum() {
     
        Light[] allLight = Light.values();
     
        for(Light aLight : allLight) {
     
          System.out.println("当前灯name:"+ aLight.name());
     
          System.out.println("当前灯ordinal:"+ aLight.ordinal());
     
          System.out.println("当前灯:"+ aLight);
     
        }
     
      }
     
      /**
       *
       * 演示EnumMap的使用,EnumMap跟HashMap的使用差不多,只不过key要是枚举类型
       */
     
      privatestaticvoid testEnumMap() {
     
        // 1.演示定义EnumMap对象,EnumMap对象的构造函数需要参数传入,默认是key的类的类型
     
        EnumMap<Light, String> currEnumMap = newEnumMap<Light, String>(
     
        Light.class);
     
        currEnumMap.put(Light.RED,"红灯");
     
        currEnumMap.put(Light.GREEN,"绿灯");
     
        currEnumMap.put(Light.YELLOW,"黄灯");
     
        // 2.遍历对象
     
        for(Light aLight : Light.values()) {
     
          System.out.println("[key="+ aLight.name() + ",value="
     
          + currEnumMap.get(aLight) + "]");
     
        }
     
      }
     
      /**
       *
       * 演示EnumSet如何使用,EnumSet是一个抽象类,获取一个类型的枚举类型内容<BR/>
       *
       * 可以使用allOf方法
       */
     
      privatestaticvoid testEnumSet() {
     
        EnumSet<Light> currEnumSet = EnumSet.allOf(Light.class);
     
        for(Light aLightSetElement : currEnumSet) {
     
          System.out.println("当前EnumSet中数据为:"+ aLightSetElement);
     
        }
     
      }
     
    }

    执行结果如下:

    演示枚举类型的遍历 ......

    当前灯name:RED

    当前灯ordinal:0

    当前灯:1

    当前灯name:GREEN

    当前灯ordinal:1

    当前灯:3

    当前灯name:YELLOW

    当前灯ordinal:2

    当前灯:2

    演示EnmuMap对象的使用和遍历.....

    [key=RED,value=红灯]

    [key=GREEN,value=绿灯]

    [key=YELLOW,value=黄灯]

    演示EnmuSet对象的使用和遍历.....

    当前EnumSet中数据为:1

    当前EnumSet中数据为:3

    当前EnumSet中数据为:2

    四、 通常定义常量方法和枚举定义常量方法区别

    以下内容可能有些无聊,但绝对值得一窥

    1. 代码:

    ?
    1
    2
    3
    4
    5
    6
    7
    publicclassState {
     
    publicstaticfinal int ON = 1;
     
    publicstaticfinal Int OFF= 0;
     
    }

    有什么不好了,大家都这样用了很长时间了,没什么问题啊。

    首先,它不是类型安全的。你必须确保是int

    其次,你还要确保它的范围是0和1

    最后,很多时候你打印出来的时候,你只看到 1 和0 ,

    但其没有看到代码的人并不知道你的企图,抛弃你所有旧的public static final常量

    2. 可以创建一个enum类,把它看做一个普通的类。除了它不能继承其他类了。(java是单继承,它已经继承了Enum),

    可以添加其他方法,覆盖它本身的方法

    3. switch()参数可以使用enum了

    4. values()方法是编译器插入到enum定义中的static方法,所以,当你将enum实例向上转型为父类Enum是,values()就不可访问了。解决办法:在Class中有一个getEnumConstants()方法,所以即便Enum接口中没有values()方法,我们仍然可以通过Class对象取得所有的enum实例

    5. 无法从enum继承子类,如果需要扩展enum中的元素,在一个接口的内部,创建实现该接口的枚举,以此将元素进行分组。达到将枚举元素进行分组。

    6. 使用EnumSet代替标志。enum要求其成员都是唯一的,但是enum中不能删除添加元素。

    7. EnumMap的key是enum,value是任何其他Object对象。

    8. enum允许程序员为eunm实例编写方法。所以可以为每个enum实例赋予各自不同的行为。

    9. 使用enum的职责链(Chain of Responsibility) .这个关系到设计模式的职责链模式。以多种不同的方法来解决一个问题。然后将他们链接在一起。当一个请求到来时,遍历这个链,直到链中的某个解决方案能够处理该请求。

    10. 使用enum的状态机。

    11. 使用enum多路分发。

    enum 对象的常用方法介绍

    int compareTo(E o) 
              比较此枚举与指定对象的顺序。

    Class<E> getDeclaringClass() 
              返回与此枚举常量的枚举类型相对应的 Class 对象。

    String name() 
              返回此枚举常量的名称,在其枚举声明中对其进行声明。

    int ordinal() 
              返回枚举常量的序数(它在枚举声明中的位置,其中初始常量序数为零)。

    String toString()

               返回枚举常量的名称,它包含在声明中。

    static <T extends Enum<T>> T valueOf(Class<T> enumType, String name) 
              返回带指定名称的指定枚举类型的枚举常量。


  10. eg:public class EnumTest {   public static void main(String[] args) {      forEnum();  //        useEnumInJava();     }      /**      * 循环枚举,输出ordinal属性;若枚举有内部属性,则也输出。(说的就是我定义的TYPE类型的枚举的typeName属性)      */     private static void forEnum(){      for(SimpleEnum simpleEnum : SimpleEnum.values()){         System.out.println(simpleEnum +"**"+simpleEnum.name() + "  ordinal  " + simpleEnum.ordinal());      }      System.out.println("------------------");          for (TYPE type : TYPE.values()) {             System.out.println("type = " + type + "    type.name = " + type.name() + "   code = " + type.getCode() + "   msg = "   + type.getMsg()+"   ordinal = " + type.ordinal());//            System.out.println("type = " + type + "    type.name = " + type.name() + "   typeName = " + type.getTypeName() + "   ordinal = " + type.ordinal());          }     }      /**      * 在Java代码使用枚举      */  // private static void useEnumInJava(){//    String typeName = "f5";//    TYPE type = TYPE.fromTypeName(typeName);//    if (TYPE.BALANCE.equals(type)) {  //            System.out.println("根据字符串获得的枚举类型实例跟枚举常量一致");  //        } else {  //            System.out.println("大师兄代码错误");  //        }  // }   /**    * 季节枚举(不带参数的枚举常量)这个是最简单的枚举使用实例      * Ordinal 属性,对应的就是排列顺序,从0开始。     */   private enum SimpleEnum{      SPRING,SUMMER,AUTUMN,WINTER   }      /**      * 常用类型(带参数的枚举常量,这个只是在书上不常见,实际使用还是很多的,看懂这个,使用就不是问题啦。)      */     private enum TYPE{      FIREWALL(2000,"firewall"),        SECRET(3000,"secretMac"),        success(1000, "操作成功");//        BALANCE("f5");      //    private String typeName;      //    TYPE(String typeName){//       this.typeName = typeName;//    }      private int code;       private String msg;       TYPE(int code, String msg) {           this.code = code;           this.msg = msg;       }       public int getCode() {           return code;       }       public String getMsg() {           return msg;       }       /**          * 根据类型的名称,返回类型的枚举实例。          *          * @param typeName 类型名称          */ //    public static TYPE fromTypeName(String typeName){//       for(TYPE type : TYPE.values()){//          if(type.getTypeName().equals(typeName)){//             return type;//          }//       }//       return null;//    }      //    public String getTypeName(){//       return this.typeName;//    }   }   }