为什么要有枚举
问题:要定义星期几或性别的变量,该怎么定义?假设用1-7分别表示星期一到星期日,但有的人可能会写成int weekday=0;
枚举就是要让某个类型的变量的取值只能是若干个固定值中的一个,否则编译器就会报错。枚举可以让编译器在编译时就可以控制源程序中填写的非法值,普通变量的方式在开发阶段无法实现这一目标。
用普通类如何实现枚举功能,定义一个Weekday的类来模拟枚举功能。
1. 构造函数私有化
2. 每个元素跟别用一个共有的静态成员变量表示
3. 可以有若干共有方法或抽象方法,例如,要提供nextDay方法必须是抽象的。采用抽象方法定义nextDay就像大量的if...else语句转换成了一个个独立的类。
package cn.cast.day1;
public abstract class WeekDay {
private WeekDay(){}
public final static WeekDay SUN=new WeekDay(){
public WeekDay nextDay(){
return MON;
}
};
public final static WeekDay MON=new WeekDay(){
public WeekDay nextDay(){
return SUN;
}
};
public abstract WeekDay nextDay();
/*public WeekDay nextDay(){
if(this==SUN){
return MON;
}else{
return SUN;
}
}*/
public String toString(){
return this==SUN?"SUN":"MON";
}
}
package cn.cast.day1;
public class EnumTest {
public static void main(String[] args){
WeekDay weekDay=WeekDay.MON;
System.out.println(weekDay.nextDay());
}
}
结果:
SUN
枚举的基本应用
举例:定义一个Weekday的枚举。
扩展:枚举类的values,valueOf,name,toString,ordinal等方法
总结:枚举是一种特殊的类,其中的每一个元素都是该类的一个实例对象,例如可以调用WeekDay.SUN.getClass().getName和WeekDay.Class.getName()
enum很像特殊的class,实际上enum声明定义的类型就是一个类。而这些类都是类库中Enum类的子类(java.lang.Enum<E>)。它们继承了这个Enum中的许多有用的方法 。
package cn.cast.day1;
public class EnumTest {
public static void main(String[] args){
WeekDay weekDay2=WeekDay.FRI;
System.out.println(weekDay2);
System.out.println(weekDay2.name());
System.out.println(weekDay2.ordinal());
System.out.println(WeekDay.valueOf("SUN"));
System.out.println(WeekDay.values().length);
}
public enum WeekDay{
SUN,MON,TUE,WED,THI,FRI,SAT;
}
}
结果:
FRI
FRI
5
SUN
7
枚举就相当于一个类,其中也可以定义构造方法、成员变量、普通方法和抽象方法。
枚举元素必须位于枚举体中的最开始的部分,枚举元素列表的后要有分号与其它成员分隔。把枚举中的成员方法或变量等放在枚举元素的前面,编译器会报错。
实现带有构造方法的枚举
1.构造函数必须定义为私有
2.如果有多个构造方法,该如何选择构造方法?
枚举元素MON和MON()的效果一样,都是调用默认的构造方法。
package cn.cast.day1;
public class EnumTest {
public static void main(String[] args){
WeekDay weekDay2=WeekDay.FRI;
System.out.println(weekDay2);
}
public enum WeekDay{
SUN(1),MON(),TUE,WED,THI,FRI,SAT;
/*构造方法必须放在枚举成员之后*/
private WeekDay(){System.out.println("first");}//必须是私有的
private WeekDay(int day){System.out.println("second");}
}
}
结果:
second
first
first
first
first
first
first
FRI
注意:
1. 枚举类中的其他方法必须放在枚举成员之后
2. 私有化构造函数
实现带抽象方法的枚举
1. 定义枚举TrafficLamp
2. 实现普通的next方法
3. 实现抽象的next方法:每个元素分别是由枚举类的子类来生成的实例对象,这些子类采用类似内部类的方式进行定义。
增加上表示时间的构造方法。
public enum TrafficLamp{
RED(30){
public TrafficLamp nextLamp(){
return GREEN;
}
},
GREEN(30){
public TrafficLamp nextLamp(){
return YELLOW;
}
},
YELLOW(3){
public TrafficLamp nextLamp(){
return RED;
}
};
public abstract TrafficLamp nextLamp();
private int time;
private TrafficLamp(int time){this.time=time;}
}
枚举只有一个成员时,就可以作为一种单例的实现方式。