java enum(枚举)使用详解 + 总结

1. 原始的接口定义常量

2. 语法(定义)

3. 遍历、switch 等常用操作

4. enum 对象的常用方法介绍

5. 给 enum 自定义属性和方法

6. EnumSet,EnumMap 的应用

7. enum 的原理分析

8. 总结

原始的接口定义常量

public  interface  IConstants {
     String MON = "Mon" ;
     String TUE = "Tue" ;
     String WED = "Wed" ;
     String THU = "Thu" ;
     String FRI = "Fri" ;
     String SAT = "Sat" ;
     String SUN = "Sun" ;
}

语法(定义)

    创建枚举类型要使用 enum 关键字,隐含了所创建的类型都是 java.lang.Enum 类的子类(java.lang.Enum 是一个抽象类)。枚举类型符合通用模式 Class Enum>,而 E 表示枚举类型的名称。枚举类型的每一个值都将映射到 protected Enum(String name, int ordinal) 构造函数中,在这里,每个值的名称都被转换成一个字符串,并且序数设置表示了此设置被创建的顺序。

package  com.hmw.test;
/**
  * 枚举测试类
  * @author 何明旺
  */
public  enum  EnumTest {
     MON, TUE, WED, THU, FRI, SAT, SUN;
}

这段代码实际上调用了7次 Enum(String name, int ordinal):

new  Enum( "MON" , 0 );
new  Enum( "TUE" , 1 );
new  Enum( "WED" , 2 );
     ... ...

遍历、switch 等常用操作

对enum进行遍历和switch的操作示例代码:

public  class  Test {
     public  static  void  main(String[] args) {
         for  (EnumTest e : EnumTest.values()) {
             System.out.println(e.toString());
         }
         
         System.out.println( "----------------我是分隔线------------------" );
         
         EnumTest test = EnumTest.TUE;
         switch  (test) {
         case  MON:
             System.out.println( "今天是星期一" );
             break ;
         case  TUE:
             System.out.println( "今天是星期二" );
             break ;
         // ... ...
         default :
             System.out.println(test);
             break ;
         }
     }
}

输出结果:

MON
TUE
WED
THU
FRI
SAT
SUN
----------------我是分隔线------------------
今天是星期二

enum 对象的常用方法介绍

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

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

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

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

String toString()

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

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

public  class  Test {
     public  static  void  main(String[] args) {
         EnumTest test = EnumTest.TUE;
         
         //compareTo(E o)
         switch  (test.compareTo(EnumTest.MON)) {
         case  - 1 :
             System.out.println( "TUE 在 MON 之前" );
             break ;
         case  1 :
             System.out.println( "TUE 在 MON 之后" );
             break ;
         default :
             System.out.println( "TUE 与 MON 在同一位置" );
             break ;
         }
         
         //getDeclaringClass()
         System.out.println( "getDeclaringClass(): "  + test.getDeclaringClass().getName());
         
         //name() 和  toString()
         System.out.println( "name(): "  + test.name());
         System.out.println( "toString(): "  + test.toString());
         
         //ordinal(), 返回值是从 0 开始
         System.out.println( "ordinal(): "  + test.ordinal());
     }
}

输出结果:

TUE 在 MON 之后
getDeclaringClass(): com.hmw.test.EnumTest
name(): TUE
toString(): TUE
ordinal(): 1

给 enum 自定义属性和方法

给 enum 对象加一下 value 的属性和 getValue() 的方法:

package  com.hmw.test;
 
/**
  * 枚举测试类
  *
  * @author 何明旺
  */
public  enum  EnumTest {
     MON( 1 ), TUE( 2 ), WED( 3 ), THU( 4 ), FRI( 5 ), SAT( 6 ) {
         @Override
         public  boolean  isRest() {
             return  true ;
         }
     },
     SUN( 0 ) {
         @Override
         public  boolean  isRest() {
             return  true ;
         }
     };
 
     private  int  value;
 
     private  EnumTest( int  value) {
         this .value = value;
     }
 
     public  int  getValue() {
         return  value;
     }
 
     public  boolean  isRest() {
         return  false ;
     }
}
public  class  Test {
     public  static  void  main(String[] args) {
         System.out.println( "EnumTest.FRI 的 value = "  + EnumTest.FRI.getValue());
     }
}

输出结果:

EnumTest.FRI 的 value = 5

EnumSet,EnumMap 的应用

public  class  Test {
     public  static  void  main(String[] args) {
         // EnumSet的使用
         EnumSet weekSet = EnumSet.allOf(EnumTest. class );
         for  (EnumTest day : weekSet) {
             System.out.println(day);
         }
 
         // EnumMap的使用
         EnumMap weekMap = new  EnumMap(EnumTest. class );
         weekMap.put(EnumTest.MON, "星期一" );
         weekMap.put(EnumTest.TUE, "星期二" );
         // ... ...
         for  (Iterator> iter = weekMap.entrySet().iterator(); iter.hasNext();) {
             Entry entry = iter.next();
             System.out.println(entry.getKey().name() + ":"  + entry.getValue());
         }
     }
}

原理分析

        enum 的语法结构尽管和 class 的语法不一样,但是经过编译器编译之后产生的是一个class文件。该class文件经过反编译可以看到实际上是生成了一个类,该类继承了java.lang.Enum。EnumTest 经过反编译(javap com.hmw.test.EnumTest 命令)之后得到的内容如下:

public  class  com.hmw.test.EnumTest extends  java.lang.Enum{
     public  static  final  com.hmw.test.EnumTest MON;
     public  static  final  com.hmw.test.EnumTest TUE;
     public  static  final  com.hmw.test.EnumTest WED;
     public  static  final  com.hmw.test.EnumTest THU;
     public  static  final  com.hmw.test.EnumTest FRI;
     public  static  final  com.hmw.test.EnumTest SAT;
     public  static  final  com.hmw.test.EnumTest SUN;
     static  {};
     public  int  getValue();
     public  boolean  isRest();
     public  static  com.hmw.test.EnumTest[] values();
     public  static  com.hmw.test.EnumTest valueOf(java.lang.String);
     com.hmw.test.EnumTest(java.lang.String, int , int , com.hmw.test.EnumTest);
}

所以,实际上 enum 就是一个 class,只不过 java 编译器帮我们做了语法的解析和编译而已。

总结

    可以把 enum 看成是一个普通的 class,它们都可以定义一些属性和方法,不同之处是:enum 不能使用 extends 关键字继承其他类,因为 enum 已经继承了 java.lang.Enum(java是单一继承)。但是enum可以实现接口。


用法一:常量

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. }  


/**
 * 枚举
 */
enum enumFactory implements Runnable{
front("前"){
@Override
public void run() {
System.out.println("我是前");
}
},back("后"){
@Override
public void run() {
System.out.println("我是后");
}
},left("左"){
@Override
public void run() {
System.out.println("我是左");

}
},right("右"){
@Override
public void run() {
System.out.println("我是右");

}
};
private String s;
private enumFactory(String s) {
this.s=s;
}

 

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

 

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. }  

 

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

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



你可能感兴趣的:(java)