枚舉類的使用情景
在一些情況下,一個類的對象是有限而且固定的,常見的例子如指南針的方向(EAST、SOUTH、WEST、NORTH)以及一個星期中的天數(shù),為了表示這些事物,一般的做法是定義相應(yīng)數(shù)量的靜態(tài)常量,例如:
public static final int DIRECTION_EAST = 0;
public static final int DIRECTION_SOUTH = 1;
public static final int DIRECTION_WEST = 2;
public static final int DIRECTION_NORTH = 3
但是這種做法有如下缺點:
- 類型不安全,每個靜態(tài)常量實際上都可以當(dāng)作一個 int 型進(jìn)行使用
- 沒有命名空間,使用方向的時候,還需要在每個方向的前面加上 DIRECTION 前綴
- 打印輸出的意義不明確,輸出每個季節(jié)的時候?qū)嶋H上輸出的是一個 int 型數(shù)據(jù)
所以對于這種情況而言,Java 5 新增了 enum 關(guān)鍵字用于定義枚舉類。一個簡單的枚舉類定義如下:
public enum SeasonEnum {
SPRING, SUMMER, FALL, WINTER;
}
枚舉類和普通類的區(qū)別
枚舉類實際上是一種特殊的類,相較之普通的類而言,其主要的區(qū)別在于:
- 由于所有枚舉類都是 Enum 類的子類,所以枚舉類不能夠繼承其它的類
- 使用 enum 定義、非抽象的枚舉類默認(rèn)使用 final 修飾符,所以非抽象的枚舉類不能夠派生子類
- 定義的枚舉類可以為空,但是一旦定義成員,必須在首行列出所有可能的實例值,而且默認(rèn)用 public static final 進(jìn)行修飾
- 枚舉類的構(gòu)造器只能夠使用 private 修飾符進(jìn)行修飾
枚舉類的成員
枚舉類的成員可以是成員變量、方法、構(gòu)造器、初始化塊以及內(nèi)部類和接口。
初始化塊、內(nèi)部類和接口以及靜態(tài)變量很少使用,個人覺得不使用初始化塊的原因在于枚舉類的大部分成員變量是用 private final 進(jìn)行修飾,而且對于不同的實例表示變量值一般不同,這就需要用到有參的構(gòu)造函數(shù)進(jìn)行傳值,那么這個時候使用初始化塊對變量進(jìn)行初始化會造成沖突。不使用內(nèi)部類和接口以及靜態(tài)變量的原因可能是破壞了枚舉類的作用,枚舉類是用于列舉出該類所有可能的實例,如果定義了靜態(tài)變量,在使用枚舉類.枚舉名訪問枚舉值的時候,會出現(xiàn) 4 個季節(jié)有 5 個選擇的情況。
接下來說說成員變量、方法、構(gòu)造器可以使用的修飾符:
- 成員變量:private、protected、public 都可以使用,但是為了使枚舉類成為不可變類,所以一般使用 private final 進(jìn)行修飾
- 構(gòu)造器:只能 private,省略了系統(tǒng)給你補(bǔ)上
- 方法:private、protected、public 都可以
枚舉類的常用方法
枚舉類作為 Enum 的子類,其方法結(jié)構(gòu)是:

接下來分別說說常用的方法及其作用:
- values():返回枚舉類的所有實例,一般用于遍歷
for (SeasonEnum season : SeasonEnum.values()) {
System.out.println(season);
}
輸出結(jié)果:
SPRING
SUMMER
FALL
WINTER
- name() :返回枚舉實例的名稱,大多時候應(yīng)該使用 toString() 方法,因為 toString() 返回更加友好的名稱
System.out.println(SeasonEnum.SPRING.name());
輸出結(jié)果:
SPRING
- oridinal():返回枚舉值在枚舉類中的索引值(就是枚舉值在枚舉類中聲明的位置)
System.out.println(SeasonEnum.SPRING.oridinal());
輸出結(jié)果:
0
toString():返回枚舉值常量的名稱,與 name 方法類似
valueOf(): 用于返回枚舉類中指定的枚舉值
System.out.println(Enum.valueOf(SeasonEnum.class, "SUMMER"));
輸出結(jié)果:
SUMMER
枚舉類的使用
- 定義了成員變量、方法以及構(gòu)造器的枚舉類,枚舉類最好是設(shè)計成不可變的類,所以一種比較好的實現(xiàn)方式是:
public enum SeasonEnum {
SPRING("春"), SUMMER("夏"), FALL("秋"), WINTER("冬");
private final String name;
SeasonEnum(String name) {
this.name = name;
}
public String getName() {
return name;
}
}
通過這種方式創(chuàng)建的枚舉類,一旦列出枚舉值,其 name 字段就再不不可更改,從而保證了枚舉類不可變。
一旦為枚舉類定義了帶參的構(gòu)造器,列出枚舉值的時候就必須傳入對應(yīng)的參數(shù)
- 實現(xiàn)接口的枚舉類
定義一個輸出季節(jié)的接口:
public interface SeasonInterface {
void printSeason();
}
在上述枚舉類的基礎(chǔ)上實現(xiàn)接口,有兩種方式,一種和普通類實現(xiàn)接口方式,另一種則是為每個枚舉值實現(xiàn)不同的方法以表現(xiàn)出不同的特征:
public enum SeasonEnum implements SeasonInterface{
SPRING("春") {
@Override
public void printSeason() {
System.out.println("春天到了");
}
},
SUMMER("夏") {
@Override
public void printSeason() {
System.out.println("夏天到了");
}
},
FALL("秋") {
@Override
public void printSeason() {
System.out.println("秋天到了");
}
},
WINTER("冬") {
@Override
public void printSeason() {
System.out.println("冬天到了");
}
};
private final String name;
SeasonEnum(String name) {
this.name = name;
}
public String getName() {
return name;
}
}
這里的方法實現(xiàn)方式類似匿名內(nèi)部類的語法,實際上這種情況下在創(chuàng)建枚舉值的時候并不是直接創(chuàng)建的 SeasonEnum 的實例,而是創(chuàng)建的是 SeasonEnum 的匿名子類的實例,之所以能夠派生子類的原因在于抽象枚舉類是用 abstract 進(jìn)行修飾的而不是 final 。對于一個枚舉類而言只要包含了抽象方法,就會默認(rèn)使用 abstract 進(jìn)行修飾。
反編譯之后的類的聲明部分如下:
public abstract class SeasonEnum extends java.lang.Enum<SeasonEnum> implements SeasonInterface