J2SE5.0新特性之枚举类型
在传统的设计中,为了达到c、c++的枚举的功能,我们一般在类中声明几个常量:
public static final String BELLE_XISHI = “西施”;
public static final String BELLE_WANGZHAOJUN = “王昭君”;
public static final String BELLE_DIAOCHAN = “貂禅”;
public static final String BELLE_YANGGUIFEI = “杨贵妃”;
然而,这样做有些问题:
1. 不是类型安全的:因为被声明为String类型,您可以给它们赋值任意的字符串,而且,有可能两个常量的值相同。
2. 没有命名空间:为了区别与非分类,我们定义的常量以BELLE做前缀。
3. 脆弱:一旦被编译后,如果想插入常量,或者他们的顺序和值被更改,使用它们的程序需要重新编译。
4. 打印值代表的意义不明显。
现在,在新的Java语言中,已经可以使用枚举类型解决上面的问题,我们首先以一个例子来看一下枚举类型是怎么工作的。
package com.kuaff.jdk5; public class EnumShow1 { enum Belle { 西施, 貂禅, 王昭君, 杨贵妃, } public static void main(String[] args) { EnumShow1 show = new EnumShow1(); for (Belle belle : Belle.values()) { System.out.printf("美女的名字=%s%n", belle); } } }
这里定义了一个枚举类型Belle(美女)。它四个值:西施、貂禅、王昭君和杨贵妃。注意,这里的值可是中文。
通过Belle.values()方法可以得到Belle枚举类型的所有的值,它返回一个包含Belle值的数组。
这样我们就可以通过belle.getName()或者belle.toString()得到枚举值。
注意:这里用到了printf方法和Foreach类型的循环,在后面的文章中会对这些新特性进行介绍。
事实上,每个枚举值可以设置多个参数,我们将上面的例子扩充一下:
package com.kuaff.jdk5; public class EnumShow2 { enum Belle { 西施("春秋战国", "范蠡,勾践,夫差"), 王昭君("西汉", "刘?,毛延寿,呼韩邪单于"), 貂禅("三国", "吕布,董卓"), 杨贵妃("唐朝", "李隆基"); private final String empire; private final String mans; private Belle(String empire, String mans) { this.empire = empire; this.mans = mans; } } public static void main(String[] args) { EnumShow2 show = new EnumShow2(); for (Belle belle : Belle.values()) { System.out.printf("美女的名字:%s, 生活年代:%s, 美女身后的相关男人:%s%n",belle,belle.empire,belle.mans); } } }
在这个例子中,我们为每位美人都加了一个生活年代和相关的男人这两个参数。
这里必须有Belle(String empire, String mans)这样一个构造方法。对每个枚举值,可以象使用对象一样使用它,要得到它的参数,可以通过 枚举对象.参数 方式访问。如果反编译以下它的类,事实上,java编译器也是把它编译成一个类:
static class EnumShow2$Belle extends Enum
{
…
}
以上两个例子可以在Eclipse 3.1M3下编译通过,现在的Eclipse3.1M3已经对JDK5达到了97%的支持。
既然它被编译成了类,我们也会联想到,类可以有方法的,枚举类型是不是页支持方法内呢?是!
我们在扩充一下上面的例子,为每个美人增加一个显示其生平事件的方法:
package com.kuaff.jdk5; public class EnumShow3 { enum Belle { 西施("春秋战国", "范蠡,勾践,夫差") { String mainEvent() { return "为勾践,嫁夫差,与范蠡泛西湖终老"; } }, 王昭君("西汉", "刘?,毛延寿,呼韩邪单于") { String mainEvent() { return "毛延寿,不厚道,刘?不走访基层,不听群众意见,昭君和亲,青冢埋芳泽"; } }, 貂禅("三国", "吕布,董卓") { String mainEvent() { return "红颜真祸水乎?男人好色而已!"; } }, 杨贵妃("唐朝", "李隆基") { String mainEvent() { return "温泉在,佳人无,长生殿内暗伤神,却闻凤凰东度。"; } } ; private final String empire; private final String mans; private Belle(String empire, String mans) { this.empire = empire; this.mans = mans; } abstract String mainEvent(); } public static void main(String[] args) { EnumShow3 show = new EnumShow3(); for (Belle belle : Belle.values()) { System.out.printf("美女的名字:%s, 生活年代:%s, 美女身后的相关男人:%s%n",belle,belle.empire,belle.mans); System.out.printf(" 相关事件:%s%n", belle.mainEvent() ); } } }
我们为每个美人增加了一个方法String mainEvent();同时,我们也必须在此枚举类型内增加一个抽象的方法声明:abstract String mainEvent();
和类型一样,通过 枚举对象.方法(…) 来调用枚举类型的方法。
上面的例子在eclipse 3.1M3下显示有错误,因为目前的eclipse还认为抽象方法只能在抽象类中声明,没关系,在命令行中直接编译吧: javac -source 5 -target 5 -d bin src/com/kuaff/jdk5/EnumShow3.java。