结果:Mr. John Doe
转的。枚举是一个特殊的类。
枚举类型是JDK5.0的新特征 。Sun引进了一个全新的关键字enum来定义一个枚举类 。下面就是一个典型枚举类型的定义:
Java代码:
public enum Color{
RED,BLUE,BLACK,YELLOW,GREEN
}
显然,enum很像特殊的class,实际上enum声明定义的类型就是一个类 。 而这些类都是类库中Enum类的子类(java.lang.Enum) 。它们继承了这个Enum中的许多有用的方法 。我们对代码编译之后发现,编译器将enum类型单独编译成了一个字节码文件:Color.class 。
Color字节码代码
final enum hr.test.Color {
// 所有的枚举值都是类静态常量
public static final enum hr.test.Color RED;
public static final enum hr.test.Color BLUE;
public static final enum hr.test.Color BLACK;
public static final enum hr.test.Color YELLOW;
public static final enum hr.test.Color GREEN;
private static final synthetic hr.test.Color〔〕 ENUM$VALUES;
// 初始化过程,对枚举类的所有枚举值对象进行第一次初始化
static {
0 new hr.test.Color 〔1〕
3 dup
4 ldc 〔16〕 //把枚举值字符串“RED”压入操作数栈
6 iconst_0 // 把整型值0压入操作数栈
7 invokespecial hr.test.Color(java.lang.String, int) 〔17〕 //调用Color类的私有构造器创建Color对象RED
10 putstatic hr.test.Color.RED : hr.test.Color 〔21〕 //将枚举对象赋给Color的静态常量RED 。
。.. 。.. 。.. 枚举对象BLUE等与上同
102 return
};
// 私有构造器,外部不可能动态创建一个枚举类对象(也就是不可能动态创建一个枚举值) 。
private Color(java.lang.String arg0, int arg1){
// 调用父类Enum的受保护构造器创建一个枚举对象
3 invokespecial java.lang.Enum(java.lang.String, int) 〔38〕
};
public static hr.test.Color〔〕 values();
// 实现Enum类的抽象方法
public static hr.test.Color valueOf(java.lang.String arg0);
}
下面我们就详细介绍enum定义的枚举类的特征及其用法 。(后面均用Color举例)
1、Color枚举类就是class,而且是一个不可以被继承的final类 。
其枚举值(RED,BLUE. 。.)都是Color类型的类静态常量, 我们可以通过下面的方式来得到Color枚举类的一个实例:
Color c=Color.RED;
注意:这些枚举值都是public static final的,也就是我们经常所定义的常量方式,因此枚举类中的枚举值最好全部大写 。
2、即然枚举类是class,当然在枚举类型中有构造器,方法和数据域 。
但是,枚举类的构造器有很大的不同:
(1) 构造器只是在构造枚举值的时候被调用 。
Java代码:
enum Color{
RED(255,0,0),BLUE(0,0,255),BLACK(0,0,0),YELLOW(255,255,0),GREEN(0,255,0);
//构造枚举值,比如RED(255,0,0)
private Color(int rv,int gv,int bv){
this.redValue=rv;
this.greenValue=gv;
this.blueValue=bv;
}
public String toString(){ //覆盖了父类Enum的toString()
return super.toString()+“(”+redValue+“,”+greenValue+“,”+blueValue+“)”;
}
private int redValue; //自定义数据域,private为了封装 。
private int greenValue;
private int blueValue;
}
(2) 构造器只能私有private,绝对不允许有public构造器 。 这样可以保证外部代码无法新构造枚举类的实例 。这也是完全符合情理的,因为我们知道枚举值是public static final的常量而已 。 但枚举类的方法和数据域可以允许外部访问 。
Java代码:
public static void main(String args〔〕)
{
// Color colors=new Color(100,200,300); //wrong
Color color=Color.RED;
System.out.println(color); // 调用了toString()方法
}
3、所有枚举类都继承了Enum的方法,下面我们详细介绍这些方法 。
(1) ordinal()方法: 返回枚举值在枚举类种的顺序 。这个顺序根据枚举值声明的顺序而定 。
Color.RED.ordinal(); //返回结果:0
Color.BLUE.ordinal(); //返回结果:1
(2) compareTo()方法: Enum实现了java.lang.Comparable接口,因此可以比较象与指定对象的顺序 。Enum中的compareTo返回的是两个枚举值的顺序之差 。当然,前提是两个枚举值必须属于同一个枚举类,否则会抛出ClassCastException()异常 。(具体可见源代码)
Color.RED.compareTo(Color.BLUE); //返回结果 -1
(3) values()方法: 静态方法,返回一个包含全部枚举值的数组 。
Color〔〕 colors=Color.values();
for(Color c:colors){
System.out.print(c+“,”);
}//返回结果:RED,BLUE,BLACK YELLOW,GREEN,
(4) toString()方法: 返回枚举常量的名称 。
Color c=Color.RED;
System.out.println(c);//返回结果: RED
(5) valueOf()方法: 这个方法和toString方法是相对应的,返回带指定名称的指定枚举类型的枚举常量 。
Color.valueOf(“BLUE”); //返回结果: Color.BLUE
(6) equals()方法: 比较两个枚举类对象的引用 。
Java代码:
//JDK源代码:
public final boolean equals(Object other) {
return this==other;
}
4、枚举类可以在switch语句中使用 。
Java代码:
Color color=Color.RED;
switch(color){
case RED: System.out.println(“it‘s red”);break;
case BLUE: System.out.println(“it’s blue”);break;
case BLACK: System.out.println(“it‘s blue”);break;
}
怎么说呢,前三行代码知道吧,第二行就是普通的属性,第三行是构造方法,重点是第一行就是用第三行的构造方法来构造三个常量,MR(“Mr”), MR是名字Mr就是构造参数,没有为什么 就是语法,你要问的是format方法吗,format方法就算是一个方便的快捷方式 ,用于取到想要的常量的值。不知道够不够明白。
Mr. John Doe
该枚举有3个Title类型的成员,所以首先,调用了3次构造方法,为每个枚举成员进行了titile的赋值,所以,枚举成员Title.MR的title被赋值为“Mr.”
然后枚举成员Title.MR(注意:它本身就是Title类型)调用了format方法,
返回了字符串Mr. John Doe
程序这样写才能运行,将main函数加入Title中:
public enum Title
{
MR("Mr."), MRS("Mrs."), MS("Ms.");
private final String title;
private Title(String t) { title = t; }
public String format(String last, String first)
{
return title + " " + first + " " + last;
}
public static void main(String[] args) {
System.out.println(Title.MR.format("Doe", "John"));
}
}
结果是:Mr. John Doe
枚举虽然不是类,但是可以当类来使用,里面是可以定义方法的,包括Main方法,定义的方式和作用同一般的类一样