Java的枚举类型使用方法详解
1.背景
在java语言中还没有引入枚举类型之前,表示枚举类型的常用模式是声明一组具有int常量。之前我们通常利用publicfinalstatic方法定义的代码如下,分别用1表示春天,2表示夏天,3表示秋天,4表示冬天。
publicclassSeason{
publicstaticfinalintSPRING=1;
publicstaticfinalintSUMMER=2;
publicstaticfinalintAUTUMN=3;
publicstaticfinalintWINTER=4;
}
这种方法称作int枚举模式。可这种模式有什么问题呢,我们都用了那么久了,应该没问题的。通常我们写出来的代码都会考虑它的安全性、易用性和可读性。首先我们来考虑一下它的类型安全性。当然这种模式不是类型安全的。比如说我们设计一个函数,要求传入春夏秋冬的某个值。但是使用int类型,我们无法保证传入的值为合法。代码如下所示:
privateStringgetChineseSeason(intseason){
StringBufferresult=newStringBuffer();
switch(season){
caseSeason.SPRING:
result.append("春天");
break;
caseSeason.SUMMER:
result.append("夏天");
break;
caseSeason.AUTUMN:
result.append("秋天");
break;
caseSeason.WINTER:
result.append("冬天");
break;
default:
result.append("地球没有的季节");
break;
}
returnresult.toString();
}
publicvoiddoSomething(){
System.out.println(this.getChineseSeason(Season.SPRING));//这是正常的场景
System.out.println(this.getChineseSeason(5));//这个却是不正常的场景,这就导致了类型不安全问题
}
程序getChineseSeason(Season.SPRING)是我们预期的使用方法。可getChineseSeason(5)显然就不是了,而且编译很通过,在运行时会出现什么情况,我们就不得而知了。这显然就不符合Java程序的类型安全。
接下来我们来考虑一下这种模式的可读性。使用枚举的大多数场合,我都需要方便得到枚举类型的字符串表达式。如果将int枚举常量打印出来,我们所见到的就是一组数字,这是没什么太大的用处。我们可能会想到使用String常量代替int常量。虽然它为这些常量提供了可打印的字符串,但是它会导致性能问题,因为它依赖于字符串的比较操作,所以这种模式也是我们不期望的。从类型安全性和程序可读性两方面考虑,int和String枚举模式的缺点就显露出来了。幸运的是,从Java1.5发行版本开始,就提出了另一种可以替代的解决方案,可以避免int和String枚举模式的缺点,并提供了许多额外的好处。那就是枚举类型(enumtype)。接下来的章节将介绍枚举类型的定义、特征、应用场景和优缺点。
2.定义
枚举类型(enumtype)是指由一组固定的常量组成合法的类型。Java中由关键字enum来定义一个枚举类型。下面就是java枚举类型的定义。
publicenumSeason{
SPRING,SUMMER,AUTUMN,WINER;
}
3.特点
Java定义枚举类型的语句很简约。它有以下特点:
1)使用关键字enum2)类型名称,比如这里的Season3)一串允许的值,比如上面定义的春夏秋冬四季4)枚举可以单独定义在一个文件中,也可以嵌在其它Java类中。
除了这样的基本要求外,用户还有一些其他选择
5)枚举可以实现一个或多个接口(Interface)6)可以定义新的变量7)可以定义新的方法8)可以定义根据具体枚举值而相异的类
4.应用场景
以在背景中提到的类型安全为例,用枚举类型重写那段代码。代码如下:
publicenumSeason{
SPRING(1),SUMMER(2),AUTUMN(3),WINTER(4);
privateintcode;
privateSeason(intcode){
this.code=code;
}
publicintgetCode(){
returncode;
}
}
publicclassUseSeason{
/**
*将英文的季节转换成中文季节
*@paramseason
*@return
*/
publicStringgetChineseSeason(Seasonseason){
StringBufferresult=newStringBuffer();
switch(season){
caseSPRING:
result.append("[中文:春天,枚举常量:"+season.name()+",数据:"+season.getCode()+"]");
break;
caseAUTUMN:
result.append("[中文:秋天,枚举常量:"+season.name()+",数据:"+season.getCode()+"]");
break;
caseSUMMER:
result.append("[中文:夏天,枚举常量:"+season.name()+",数据:"+season.getCode()+"]");
break;
caseWINTER:
result.append("[中文:冬天,枚举常量:"+season.name()+",数据:"+season.getCode()+"]");
break;
default:
result.append("地球没有的季节"+season.name());
break;
}
returnresult.toString();
}
publicvoiddoSomething(){
for(Seasons:Season.values()){
System.out.println(getChineseSeason(s));//这是正常的场景
}
//System.out.println(getChineseSeason(5));
//此处已经是编译不通过了,这就保证了类型安全
}
publicstaticvoidmain(String[]arg){
UseSeasonuseSeason=newUseSeason();
useSeason.doSomething();
}
}
[中文:春天,枚举常量:SPRING,数据:1][中文:夏天,枚举常量:SUMMER,数据:2][中文:秋天,枚举常量:AUTUMN,数据:3][中文:冬天,枚举常量:WINTER,数据:4]
这里有一个问题,为什么我要将域添加到枚举类型中呢?目的是想将数据与它的常量关联起来。如1代表春天,2代表夏天。
5.总结
那么什么时候应该使用枚举呢?每当需要一组固定的常量的时候,如一周的天数、一年四季等。或者是在我们编译前就知道其包含的所有值的集合。Java1.5的枚举能满足绝大部分程序员的要求的,它的简明,易用的特点是很突出的。
6.用法
用法一:常量
publicenumColor{
RED,GREEN,BLANK,YELLOW
}
用法二:switch
enumSignal{
GREEN,YELLOW,RED
}
publicclassTrafficLight{
Signalcolor=Signal.RED;
publicvoidchange(){
switch(color){
caseRED:
color=Signal.GREEN;
break;
caseYELLOW:
color=Signal.RED;
break;
caseGREEN:
color=Signal.YELLOW;
break;
}
}
}
用法三:向枚举中添加新方法
publicenumColor{
RED("红色",1),GREEN("绿色",2),BLANK("白色",3),YELLO("黄色",4);
//成员变量
privateStringname;
privateintindex;
//构造方法
privateColor(Stringname,intindex){
this.name=name;
this.index=index;
}
//普通方法
publicstaticStringgetName(intindex){
for(Colorc:Color.values()){
if(c.getIndex()==index){
returnc.name;
}
}
returnnull;
}
//getset方法
publicStringgetName(){
returnname;
}
publicvoidsetName(Stringname){
this.name=name;
}
publicintgetIndex(){
returnindex;
}
publicvoidsetIndex(intindex){
this.index=index;
}
}
用法四:覆盖枚举的方法
publicenumColor{
RED("红色",1),GREEN("绿色",2),BLANK("白色",3),YELLO("黄色",4);
//成员变量
privateStringname;
privateintindex;
//构造方法
privateColor(Stringname,intindex){
this.name=name;
this.index=index;
}
//覆盖方法
@Override
publicStringtoString(){
returnthis.index+"_"+this.name;
}
}
用法五:实现接口
publicinterfaceBehaviour{
voidprint();
StringgetInfo();
}
publicenumColorimplementsBehaviour{
RED("红色",1),GREEN("绿色",2),BLANK("白色",3),YELLO("黄色",4);
//成员变量
privateStringname;
privateintindex;
//构造方法
privateColor(Stringname,intindex){
this.name=name;
this.index=index;
}
//接口方法
@Override
publicStringgetInfo(){
returnthis.name;
}
//接口方法
@Override
publicvoidprint(){
System.out.println(this.index+":"+this.name);
}
}
用法六:使用接口组织枚举
publicinterfaceFood{
enumCoffeeimplementsFood{
BLACK_COFFEE,DECAF_COFFEE,LATTE,CAPPUCCINO
}
enumDessertimplementsFood{
FRUIT,CAKE,GELATO
}
}
以上就是本文的全部内容,希望对大家的学习有所帮助。