参考:http://www.cnblogs.com/mxmbk/articles/5091999.html
http://www.cnblogs.com/hyl8218/p/5088287.html
enum 的全称为 enumeration, 是 JDK 1.5 中引入的新特性,存放在 java.lang 包中。
在Java SE5之前,我们要使用枚举类型时,通常会使用static final 定义一组int常量来标识,代码如下
public static final int MAN = 0; public static final int WOMAN = 1;
相信很多小伙伴,在实际开发中也是那么干的,既然这样已经能实现枚举的功能,为什么还要引入枚举呢?我们接着往下看当我们需要是同这组“int枚举”是代码如下
public void showSex(int sex){ switch(sex){ case MAN: System.out.println("this is a boy"); break; case WOMAN: System.out.println("this is a Girl"); break; } }
看起来这些貌似也没什么问题,但是我们知道,一个项目基本都是基于团队开发,或许只有你自己知道int类型的1代表 gril,0代表Boy。其他同事看到这个函数根本不知道其中的含义,这样的代码很明显阅读性很差,从而会造成沟通成本很高。我们接着往下看,现在你为你 的func写了很nice的注释,傻子都能看得懂(0 boy,1 gril)。但是项目组不可避免的总会出现那么一两个傻子,非要给你传个3进来,而且这样的错误编译器不会报任何错误,运行时会造成什么bug,这个只有 乔老爷知道了。所以这样的代码是极不安全的。使用枚举就能很好的避免上面的问题,接下来我们就来理一理枚举的用法。
枚举用于存储数量有限的一组固定的数据集。使用场景:上面说到的性别的表示,一年四级春夏秋冬的表示,一周七天的表示,颜色的表示等等。
语法(定义)
创建枚举类型要使用 enum 关键字,隐含了所创建的类型都是 java.lang.Enum 类的子类(java.lang.Enum 是一个抽象类)。枚举类型符合通用模式 Class Enum
,而 E
表示枚举类型的名称。枚举类型的每一个值都将映射到 protected Enum(String name, int ordinal)
构造函数中,在这里,每个值的名称都被转换成一个字符串,并且序数设置表示了此设置被创建的顺序。
package
com.hmw.test;
/**
* 枚举测试类
* @author 何明旺
*/
public
enum
EnumTest {
MON, TUE, WED, THU, FRI, SAT, SUN;
}
|
这段代码实际上调用了7次 Enum(String name, int ordinal):
new
Enum
"MON"
,
0
);
new
Enum
"TUE"
,
1
);
new
Enum
"WED"
,
2
);
... ...
|
遍历、switch 等常用操作
对enum进行遍历和switch的操作示例代码:
public
class
Test {
public
static
void
main(String[] args) {
for
(EnumTest e : EnumTest.values()) {
System.out.println(e.toString());
}
System.out.println(
"----------------我是分隔线------------------"
);
EnumTest test = EnumTest.TUE;
switch
(test) {
case
MON:
System.out.println(
"今天是星期一"
);
break
;
case
TUE:
System.out.println(
"今天是星期二"
);
break
;
// ... ...
default
:
System.out.println(test);
break
;
}
}
}
|
输出结果:
MON
TUE
WED
THU
FRI
SAT
SUN
----------------我是分隔线------------------
今天是星期二
|
enum 对象的常用方法介绍
int
compareTo(E o)
比较此枚举与指定对象的顺序。
Class
getDeclaringClass()
返回与此枚举常量的枚举类型相对应的 Class 对象。
String
name()
返回此枚举常量的名称,在其枚举声明中对其进行声明。
int
ordinal()
返回枚举常量的序数(它在枚举声明中的位置,其中初始常量序数为零)。
String
toString()
返回枚举常量的名称,它包含在声明中。
static
valueOf(Class
返回带指定名称的指定枚举类型的枚举常量。
public
class
Test {
public
static
void
main(String[] args) {
EnumTest test = EnumTest.TUE;
//compareTo(E o)
switch
(test.compareTo(EnumTest.MON)) {
case
-
1
:
System.out.println(
"TUE 在 MON 之前"
);
break
;
case
1
:
System.out.println(
"TUE 在 MON 之后"
);
break
;
default
:
System.out.println(
"TUE 与 MON 在同一位置"
);
break
;
}
//getDeclaringClass()
System.out.println(
"getDeclaringClass(): "
+ test.getDeclaringClass().getName());
//name() 和 toString()
System.out.println(
"name(): "
+ test.name());
System.out.println(
"toString(): "
+ test.toString());
//ordinal(), 返回值是从 0 开始
System.out.println(
"ordinal(): "
+ test.ordinal());
}
}
|
输出结果:
TUE 在 MON 之后
getDeclaringClass(): com.hmw.test.EnumTest
name(): TUE
toString(): TUE
ordinal(): 1
|
给 enum 自定义属性和方法
给 enum 对象加一下 value 的属性和 getValue() 的方法:
package
com.hmw.test;
/**
* 枚举测试类
*
* @author 何明旺
*/
public
enum
EnumTest {
MON(
1
), TUE(
2
), WED(
3
), THU(
4
), FRI(
5
), SAT(
6
) {
@Override
public
boolean
isRest() {
return
true
;
}
},
SUN(
0
) {
@Override
public
boolean
isRest() {
return
true
;
}
};
private
int
value;
private
EnumTest(
int
value) {
this
.value = value;
}
public
int
getValue() {
return
value;
}
public
boolean
isRest() {
return
false
;
}
}
|
public
class
Test {
public
static
void
main(String[] args) {
System.out.println(
"EnumTest.FRI 的 value = "
+ EnumTest.FRI.getValue());
}
}
|
输出结果:
EnumTest.FRI 的 value = 5
|
EnumSet,EnumMap 的应用
public
class
Test {
public
static
void
main(String[] args) {
// EnumSet的使用
EnumSet
class
);
for
(EnumTest day : weekSet) {
System.out.println(day);
}
// EnumMap的使用
EnumMap
new
EnumMap(EnumTest.
class
);
weekMap.put(EnumTest.MON,
"星期一"
);
weekMap.put(EnumTest.TUE,
"星期二"
);
// ... ...
for
(Iterator
Entry
System.out.println(entry.getKey().name() +
":"
+ entry.getValue());
}
}
}
|
原理分析
enum 的语法结构尽管和 class 的语法不一样,但是经过编译器编译之后产生的是一个class文件。该class文件经过反编译可以看到实际上是生成了一个类,该类继承了java.lang.Enum
public
class
com.hmw.test.EnumTest
extends
java.lang.Enum{
public
static
final
com.hmw.test.EnumTest MON;
public
static
final
com.hmw.test.EnumTest TUE;
public
static
final
com.hmw.test.EnumTest WED;
public
static
final
com.hmw.test.EnumTest THU;
public
static
final
com.hmw.test.EnumTest FRI;
public
static
final
com.hmw.test.EnumTest SAT;
public
static
final
com.hmw.test.EnumTest SUN;
static
{};
public
int
getValue();
public
boolean
isRest();
public
static
com.hmw.test.EnumTest[] values();
public
static
com.hmw.test.EnumTest valueOf(java.lang.String);
com.hmw.test.EnumTest(java.lang.String,
int
,
int
, com.hmw.test.EnumTest);
}
|
所以,实际上 enum 就是一个 class,只不过 java 编译器帮我们做了语法的解析和编译而已。
总结
使用enum的好处主要在于很好的阅读性和安全性。我们基本上可以把枚举类当成一个常规的Java类,可以往其中添加新的方法, 包括抽象方法甚至main方法。 不同之处是:enum 不能使用 extends 关键字继承其他类,因为 enum 已经继承了 java.lang.Enum(java是单一继承);enum也不能被其它类继承。