谈谈JDK1.5新特性之枚举数据类型

在没有枚举之前,我们想列举一些相关的常量,我们会采用如下的方式:
1
2
3
4
interface ActionInterface {
    public static final int RIGHT = 0 ;
    public static final int LEFT = 1 ;
}

  然后在某个类似于下面的方法中,使用这些常量:

?
1
2
3
4
5
6
7
8
9
10
11
12
public void playWithInterface( int num) {
    switch (num) {
    case ActionInterface.RIGHT:
        System.out.println( "RIGHT" );
        break ;
    case ActionInterface.LEFT:
        System.out.println( "LEFT" );
        break ;
    default :
        System.out.println( "Default" );
    }
}

  不知道你注意了没,在这样做的时候,请记住这类常量是 Java 中 int 类型的常量,这意味着该方法可以接受任何 int 类型的值,即使它和之前的接口中定的所有常量都不对应。因此,您需要检测上界和下界,在出现无效值的时候,你只能依赖switch中的default或者其他的处理手段,而且,如果后来又添加另外一个常量的时候,那么你就必须改变很多相关联代码中的上界,下界处理程序,才能接受这个新值。

有了枚举之后:

  你应该对于这种情况,总是使用枚举,比如:

?
1
2
3
enum ActionEnum{
    RIGHT,LEFT
}

  然后在某个类似于下面的方法中,使用它们:

?
1
2
3
4
5
6
7
8
9
10
11
12
public void playWithEnum(ActionEnum actionEnum) {
    switch (actionEnum) {
    case RIGHT:
        System.out.println( "RIGHT" );
        break ;
    case LEFT:
        System.out.println( "LEFT" );
        break ;
    default :
        System.out.println( "Default" );
    }
}

  这样,你就不必费很大的心思去检查之前的一些问题,而且即便需求修改之后,添加了一些相关的变量,你只需要修改ActionEnum就行,其余代码都不需要修改。

  需要注意的是,在上面的switch中,我们使用的是case RIGHT... 而不是case ActionEnum.RIGHT(这样会出错),原因是枚举重写了ToString(),也就是说枚举变量的值是不带前缀的。

上面所说的,可能并不能完全显示出枚举类型的优势,但是确实我们经常采用的形式。下面我们来尽可能的认识一下枚举:

?
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
package com.java;
public class EnumDemo {
    /**
     * 最简单的枚举示例
     * */
    public enum SimpleEnum {
        UP, DOWN
    }
   
    /**
     * 带有属性和方法的枚举示例
     * 注意:枚举本身也是类,可以象一般的类一样为它添加静态或者非静态的属性或方法
     *        但是枚举列表必须写在最前面,否则编译出错。比如本例中的UP,DOWN
     * */
    public enum EnumWithFuncAndAttr {
        UP, DOWN;
        private static final String description = "这个是一个有方法和属性的枚举示例" ;
        public String getDescription() {
            return description;
        }
    }
   
    /**
     * 带有构造函数的枚举示例
     * 注意:枚举可以有构造方法,通过括号赋值。如果不赋值,那么就不能编写构造方法,否则出错。
     * 构造方法只能是private的或者默认的。而不能是public以及protected,否则出错。这样做可以保证客户代码没有办法新建一个enum的实例
     * */
    public enum EnumWithConstructor{
        UP( "hello" ),DOWN( "java" );
        private final String value;
        String getValue(){
            return value;
        }
        EnumWithConstructor(String value){
            this .value=value;
        }
    }
   
    public static void main(String[] args) {
        System.out.println(SimpleEnum.values().length);
        System.out.println(SimpleEnum.UP);
        System.out.println(SimpleEnum.valueOf( "UP" ));
        for (EnumWithConstructor item : EnumWithConstructor.values()) {
            System.out.println(item+ " " +item.getValue());
        }
        System.out.println(SimpleEnum.UP.ordinal());
    }
}

 在没有enum类型之前,我们是怎样做的,我们这样做的基本原理和enum类型的底层原理是基本一致的。

如:

package com.crazy.numeration;

import org.junit.Test;

public class EnumTest {

//使用单元测试进行测试结果

@Test
public void test() {
  printScore(Score.A);
  printScore2(Score.B);
  printScore3(Score2.B);
  printScore4(Score2.B);
}

//Score是传统方法显示enum效果

public void printScore(Score sc) {
  System.out.println(sc.getValue());
}

public void printScore2(Score sc) {
  System.out.println(sc);
}

//Score2是enum类型

public void printScore3(Score2 sc) {
  System.out.println(sc);
}

public void printScore4(Score2 sc) {
  System.out.println(sc.toString());
}

@Test
public void testEnumValues() {
  for (Score2 s : Score2.values()) {
   System.out.println(s);
  }
}

@Test
public void testScoreValues() {
  System.out.println("testScoreValuestestScoreValues");
  for (Score s : Score.values()) {
   System.out.println(s);
  }
  System.out.println("testScoreValuestestScoreValues");
}

@Test
public void testNumValueOf() {
  System.out.println(Score2.valueOf("A"));
}

@Test
public void testScoreValueOf() {
  System.out.println(Score.valueOf("B"));
}

@Test
public void testEnumCompareTo() {
  System.out.println(Score2.A.compareTo(Score2.C));
  System.out.println(Score2.B.compareTo(Score2.C));
  System.out.println(Score2.C.compareTo(Score2.C));
  System.out.println(Score2.D.compareTo(Score2.C));
  System.out.println(Score2.E.compareTo(Score2.C));
}

@Test
public void testScoreCompareTo() {
  System.out.println(Score.A.compareTo(Score.C));
  System.out.println(Score.B.compareTo(Score.C));
  System.out.println(Score.C.compareTo(Score.C));
  System.out.println(Score.D.compareTo(Score.C));
  System.out.println(Score.E.compareTo(Score.C));
}

@Test
public void testEnumName(){
  System.out.println(Score2.A.name());
}

@Test
public void testScoreName(){
  System.out.println(Score2.A.name());
}
}

//说enum类型是一个特殊的类,原理也是这样实现的

class Score {
private String value;
private int index;

//构造方法必须是私有的或者是默认的,不能是protected或者public的

private Score() {
}

//带参数的构造方法,所以enum类型中的枚举值的时候也是需要带参数的

private Score(int index, String value) {
  this.value = value;
  this.index = index;
}

public String getValue() {
  return value;
}

public Score get(int index) {
  switch (index) {
  case 0:
   return Score.A;
  case 1:
   return Score.B;
  case 2:
   return Score.C;
  case 3:
   return Score.D;
  case 4:
   return Score.E;
  default:
   return Score.F;
  }
}

//相当与enum类型中的枚举值,枚举值都必须是public static final的,这就是有效的防止了在客户端使用new enum情况的出现,正是这样,我们使用枚举类型,也不用考虑太多的边界检测问题。

public static final Score A = new Score(0, "90-100");
public static final Score B = new Score(1, "80-89");
public static final Score C = new Score(2, "70-79");
public static final Score D = new Score(3, "60-69");
public static final Score E = new Score(4, "0-59");
public static final Score F = new Score(5, "over");

//重写了toString()方法

@Override
public String toString() {
  switch (value) {
  case "90-100":
   return "A";
  case "80-89":
   return "B";
  case "70-79":
   return "C";
  case "60-69":
   return "D";
  case "0-59":
   return "E";
  default:
   return null;
  }
}

public static Score[] values() {
  Score[] scores = { A, B, C, D, E };
  return scores;
}

public static Score valueOf(String str) {
  switch (str) {
  case "A":
   return A;
  case "B":
   return B;
  case "C":
   return C;
  case "D":
   return D;
  case "E":
   return E;
  default:
   return null;
  }

}

public int compareTo(Object obj) {
  Score other = (Score) obj;
  Score self = this;
  if (self.getClass() != other.getClass())
   throw new ClassCastException();
  return self.index - other.index;
}

public String name() {
  switch (value) {
  case "90-100":
   return "A";
  case "80-89":
   return "B";
  case "70-79":
   return "C";
  case "60-69":
   return "D";
  case "0-59":
   return "E";
  default:
   return null;
  }
}

}

 总结:

  1.枚举本身就是一个类。

  2.它不能有public的构造函数,这样做可以保证客户代码没有办法新建一个enum的实例。    

  3..所有枚举值都是public static final的。注意这一点只是针对于枚举值,我们可以和在普通类里面定义变量一样定义其它任何类型的非枚举变量,这些变量可以用任何你想用的修饰符。    

  4.Enum默认实现了java.lang.Comparable接口。    

  5.Enum覆载了了toString方法,因此我们如果调用Color.Blue.toString()默认返回字符串”Blue”.

  6.Enum提供了一个valueOf方法,这个方法和toString方法是相对应的。调用valueOf(“Blue”)将返回Color.Blue.因此我们在自己重写toString方法的时候就要注意到这一点,一般来说应该相对应地重写valueOf方法。    

  7.Enum还提供了values方法,这个方法使你能够方便的遍历所有的枚举值

  8.Enum还有一个oridinal的方法,这个方法返回枚举值在枚举类种的顺序。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值