论坛首页 Java企业应用论坛

【解惑】理解java枚举类型

浏览 36490 次
该帖已经被评为良好帖
作者 正文
   发表时间:2009-09-26   最后修改:2010-09-06

枚举类型是JDK5.0的新特征。Sun引进了一个全新的关键字enum来定义一个枚举类。下面就是一个典型枚举类型的定义:

public enum Color{
    RED,BLUE,BLACK,YELLOW,GREEN
}

显然,enum很像特殊的class,实际上enum声明定义的类型就是一个类。 而这些类都是类库中Enum类的子类(java.lang.Enum<E>)。它们继承了这个Enum中的许多有用的方法。我们对代码编译之后发现,编译器将enum类型单独编译成了一个字节码文件:Color.class。

final enum hr.test.Color {
  
 // 所有的枚举值都是类静态常量
 public static final enum hr.test.Color RED;
 public static final enum hr.test.Color BLUE;
 public static final enum hr.test.Color BLACK;
 public static final enum hr.test.Color YELLOW;
 public static final enum hr.test.Color GREEN;
 
private static final synthetic hr.test.Color[] ENUM$VALUES;
  
  // 初始化过程,对枚举类的所有枚举值对象进行第一次初始化
 static {
       0  new hr.test.Color [1] 
      3  dup
      4  ldc <String "RED"> [16] //把枚举值字符串"RED"压入操作数栈
      6  iconst_0  // 把整型值0压入操作数栈
      7  invokespecial hr.test.Color(java.lang.String, int) [17] //调用Color类的私有构造器创建Color对象RED
     10  putstatic hr.test.Color.RED : hr.test.Color [21]  //将枚举对象赋给Color的静态常量RED。
      .........  枚举对象BLUE等与上同
    102  return
};
  
  // 私有构造器,外部不可能动态创建一个枚举类对象(也就是不可能动态创建一个枚举值)。
 private Color(java.lang.String arg0, int arg1){
     // 调用父类Enum的受保护构造器创建一个枚举对象
     3  invokespecial java.lang.Enum(java.lang.String, int) [38]
};
 
 public static hr.test.Color[] values();
  
   // 实现Enum类的抽象方法  
  public static hr.test.Color valueOf(java.lang.String arg0);
}
 

下面我们就详细介绍enum定义的枚举类的特征及其用法。(后面均用Color举例)

1、Color枚举类就是class,而且是一个不可以被继承的final类。其枚举值(RED,BLUE...)都是Color类型的类静态常量, 我们可以通过下面的方式来得到Color枚举类的一个实例:
                                                         Color c=Color.RED;
注意:这些枚举值都是public static final的,也就是我们经常所定义的常量方式,因此枚举类中的枚举值最好全部大写。

2、即然枚举类是class,当然在枚举类型中有构造器,方法和数据域。但是,枚举类的构造器有很大的不同:
      (1) 构造器只是在构造枚举值的时候被调用。

enum Color{
                RED(255,0,0),BLUE(0,0,255),BLACK(0,0,0),YELLOW(255,255,0),GREEN(0,255,0);
                //构造枚举值,比如RED(255,0,0)
                private Color(int rv,int gv,int bv){
                 this.redValue=rv;
                 this.greenValue=gv;
                 this.blueValue=bv;
                }

                public String toString(){  //覆盖了父类Enum的toString()
                return super.toString()+"("+redValue+","+greenValue+","+blueValue+")";
                }
   
                private int redValue;  //自定义数据域,private为了封装。
                private int greenValue;
                private int blueValue;
 }

      (2) 构造器只能私有private,绝对不允许有public构造器。 这样可以保证外部代码无法新构造枚举类的实例。这也是完全符合情理的,因为我们知道枚举值是public static final的常量而已。 但枚举类的方法和数据域可以允许外部访问。

public static void main(String args[])
{
        // Color colors=new Color(100,200,300);  //wrong
           Color color=Color.RED;
           System.out.println(color);  // 调用了toString()方法
}   
 

3、所有枚举类都继承了Enum的方法,下面我们详细介绍这些方法。
       (1)  ordinal()方法: 返回枚举值在枚举类种的顺序。这个顺序根据枚举值声明的顺序而定。
                 Color.RED.ordinal();  //返回结果:0
                 Color.BLUE.ordinal();  //返回结果:1
       (2)  compareTo()方法: Enum实现了java.lang.Comparable接口,因此可以比较象与指定对象的顺序。Enum中的compareTo返回的是两个枚举值的顺序之差。当然,前提是两个枚举值必须属于同一个枚举类,否则会抛出ClassCastException()异常。(具体可见源代码)
                 Color.RED.compareTo(Color.BLUE);  //返回结果 -1
       (3)  values()方法: 静态方法,返回一个包含全部枚举值的数组。
                 Color[] colors=Color.values();
                 for(Color c:colors){
                        System.out.print(c+",");
                 }//返回结果:RED,BLUE,BLACK YELLOW,GREEN,
       (4)  toString()方法: 返回枚举常量的名称。
                 Color c=Color.RED;
                 System.out.println(c);//返回结果: RED
       (5)  valueOf()方法: 这个方法和toString方法是相对应的,返回带指定名称的指定枚举类型的枚举常量。
                 Color.valueOf("BLUE");   //返回结果: Color.BLUE
       (6)  equals()方法: 比较两个枚举类对象的引用。

//JDK源代码:    
public final boolean equals(Object other) {
        return this==other;
}             


4、枚举类可以在switch语句中使用。

Color color=Color.RED;
switch(color){
        case RED: System.out.println("it's red");break;
        case BLUE: System.out.println("it's blue");break;
        case BLACK: System.out.println("it's blue");break;
}

 

   发表时间:2009-09-27  
以前对于Java的enum枚举类型没怎么注意,因为写代码相对少用了点,现在LZ你的文章写得详细,真的是学习了。。。
0 请登录后投票
   发表时间:2009-09-27  
枚举我用的也很少 .
感谢楼主
0 请登录后投票
   发表时间:2009-09-27  
的确枚举类用的很少~~~
就算用到也没深入研究~~~
不错~~
0 请登录后投票
   发表时间:2009-09-27  
枚举类用的是不多,但是一旦需要用的时候却十分的方便。
0 请登录后投票
   发表时间:2009-09-27  
枚举类可以用来做单例,很方便的
0 请登录后投票
   发表时间:2009-09-27  
用的人是很少,但深入了解,在业务分析中恰当的应用应该了起到很好的作用
0 请登录后投票
   发表时间:2009-09-28  
以前也经常用枚举,但一直没注意到原来是直接继承了java/lang/Enu类,刚才去看了一下字节码确实是这样,谢谢。
0 请登录后投票
   发表时间:2009-09-28  
wangdgsc 写道
枚举类可以用来做单例,很方便的



0 请登录后投票
   发表时间:2009-09-28   最后修改:2009-09-28
看一下Effective Java第一版,然后反编译一下enum类生成的代码,更能清楚的了解和应用。
实现分析:
http://blog.csdn.net/ZangXT/archive/2008/10/29/3174741.aspx
0 请登录后投票
论坛首页 Java企业应用版

跳转论坛:
Global site tag (gtag.js) - Google Analytics