Java Enum 基本原理
1. 关于 Java Enum
学过 C/C++ 等语言的人,应该都对 Enum 类型略知一二。 Enum 一般用来表示一组相同类型的常量。如性别、日期、月份、颜色等。对这些属性用常量的好处是显而易见的,不仅可以保证单例,且比较时候可以用 ”==” 来替换 equals 。是一种好的习惯。 JDK1.5 之前没有 Enum 这个类型,那时候一般用接口常量来替代。有了 JavaEnum 之后,可以更贴近的表示这种常量。
2. 如何使用 Java Enum
简单的用法: JavaEnum 简单的用法一般用于代表一组常用常量,可用来代表一类相同类型的常量值。如:
性别:
- public enum SexEnum {
- male , female ;
- }
颜色:
- public enum Color {
- RED , BLUE,GREEN,BLACK ;
- }
枚举对象里面的值都必须是唯一的。
可以通过 Enum 类型名直接引用该常量,如 SexEnum.male,Color.RED.
复杂用法: Java 为枚举类型提供了一些内置的方法,同事枚举常量还可以有自己的方法。可以很方便的遍历枚举对象,看个下面的例子:
1.代码一 WeekDay.java :
- /**
- * @author singleAnt
- * 2009.7.29
- * 定义一个枚举类型,代表星期一到星期日的7个缩写常量
- * 同时还定义了枚举类型的属性day,该属性可以是final,也可是变量
- * 同时还定义了该枚举类型的一个方法printDay
- */
- public enum WeekDay {
- Mon("Monday"), Tue("Tuesday"), Wed("Wednesday"), Thu("Thursday"), Fri(
- "Friday"), Sat("Saturday"), Sun("Sunday");
- /**定义枚举类型自己的属性**/
- private final String day;
- private WeekDay(String day) {
- this.day = day;
- }
- /**定义枚举类型自己的方法**/
- public static void printDay(int i){
- switch(i){
- case 1: System.out.println(WeekDay.Mon); break;
- case 2: System.out.println(WeekDay.Tue);break;
- case 3: System.out.println(WeekDay.Wed);break;
- case 4: System.out.println(WeekDay.Thu);break;
- case 5: System.out.println(WeekDay.Fri);break;
- case 6: System.out.println(WeekDay.Sat);break;
- case 7: System.out.println(WeekDay.Sun);break;
- default:System.out.println("wrong number!");
- }
- }
- public String getDay() {
- return day;
- }
- }
2. 代码二 WeekDayTest.java:
- /**
- * @author singleAnt
- * 2009.7.29
- * 测试枚举类型WeekDay.
- */
- public class WeekDayTest {
- public static void main(String args[]) {
- for (WeekDay day : WeekDay.values()) {
- System.out.println(day + "====>" + day.getDay());
- }
- WeekDay.printDay(5);
- }
- }
输出结果为:
Mon====>Monday
Tue====>Tuesday
Wed====>Wednesday
Thu====>Thursday
Fri====>Friday
Sat====>Saturday
Sun====>Sunday
Fri
3. Java Enum 原理
Java Enum 类型的语法结构尽管和 java 类的语法不一样,应该说差别比较大。但是经过编译器编译之后产生的是一个 class 文件。该 class 文件经过反编译可以看到实际上是生成了一个类,该类继承了 java.lang.Enum<E>.
WeekDay (javap WeekDay 命令 之后得到的内容如下 ( 去掉了部分指令代码 ) :
- public final class WeekDay extends java.lang.Enum{
- public static final WeekDay Mon;
- public static final WeekDay Tue;
- public static final WeekDay Wed;
- public static final WeekDay Thu;
- public static final WeekDay Fri;
- public static final WeekDay Sat;
- public static final WeekDay Sun;
- static {};
- public static void printDay(int);
- public java.lang.String getDay();
- public static WeekDay[] values();
- public static WeekDay valueOf(java.lang.String);
- }
所以实际上 Enum 类型就是以 Java 类来实现的,没有什么新的特点,只不过 java 编译器帮我们做了语法的解析和编译。完全也可以自己实现。但是既然有这样方便一个东西,当然会去用了。
在大部分编程语言中,枚举类型都会是一种常用而又必不可少的数据类型,Java中当然也不会例外。然而,Java中的Enum枚举类型却有着许多你意想不到的用法,下面让我们一起来看看。
1、可以在enum中添加变量和方法
先来看一段代码示例:
public enum State { Normal("正常态", 1), Update("已更新", 2), Deleted("已删除", 3), Fired("已屏蔽", 4); // 成员变量 private String name; private int index; // 构造方法,注意:构造方法不能为public,因为enum并不可以被实例化 private State(String name, int index) { this.name = name; this.index = index; } // 普通方法 public static String getName(int index) { for (State c : State .values()) { if (c.getIndex() == index) { return c.name; } } return null; } // get set 方法 public String getName() { return name; } public void setName(String name) { this.name = name; } public int getIndex() { return index; } public void setIndex(int index) { this.index = index; } }
从上面的代码中我们可以看到,定义完枚举值,然后在其后面加上分号,接着就可以定义其他的变量、方法了。另外需要特别说明的是,enum中的构造方法不可以用public标识,这样做是为了防止用户实例化enum。
2、可以用来定义常量
先来回顾一下Java中如何定义常量吧,看下面一段代码:
public static final int normalState = 1; private static final int updateState = 2;
下面我们还可以用enum枚举来代替上面的常量定义,代码如下:
public enum State { Normal, Update, Deleted, Fired }
在Java中用enum来定义常量在语法上没有什么优势,但是enum枚举类型可以提供更多的操作功能。
3、在enum中实现接口
先来看下面一段代码:
public interface ICanReadState { void read(); String getState(); } public enum State implements ICanReadState { Normal("正常态", 1), Update("已更新", 2), Deleted("已删除", 3), Fired("已屏蔽", 4); private String name; private int index; private State(String name, int index) { this.name = name; this.index = index; } // 接口方法1 @Override public String getState() { return this.name; } // 接口方法2 @Override public void read() { System.out.println(this.index + ":" + this.name); } }
和一般的类中使用接口一样,enum枚举中同样可以继承接口,并实现接口中的所有方法,这样做的好处在于可以更方便地对枚举中的值进行排序、比较等操作,封装性更好。
总结
说白了,enum枚举类型是一个不可以被继承的final类,就以上面的State枚举为例,如果你查看enum类型的字节码,其实是State类型的类静态常量。