Java Enum用法詳解
用法一:常量
在JDK1.5 之前呻袭,我們定義常量都是: public static fianl.... 』媲鳎現(xiàn)在好了,有了枚舉,可以把相關(guān)的常量分組到一個(gè)枚舉類型里票编,而且枚舉提供了比常量更多的方法。
public enum Color {
RED, GREEN, BLANK, YELLOW
}
用法二:switch
JDK1.6之前的switch語(yǔ)句只支持int,char,enum類型卵渴,使用枚舉慧域,能讓我們的代碼可讀性更強(qiáng)。
enum Signal {
GREEN, YELLOW, RED
}
public class TrafficLight {
Signal color = Signal.RED;
public void change() {
switch (color) {
case RED:
color = Signal.GREEN;
break;
case YELLOW:
color = Signal.RED;
break;
case GREEN:
color = Signal.YELLOW;
break;
}
}
}
用法三:向枚舉中添加新方法
如果打算自定義自己的方法浪读,那么必須在enum實(shí)例序列的最后添加一個(gè)分號(hào)昔榴。而且 Java 要求必須先定義 enum 實(shí)例。
public enum Color {
RED("紅色", 1), GREEN("綠色", 2), BLANK("白色", 3), YELLO("黃色", 4);
// 成員變量
private String name;
private int index;
// 構(gòu)造方法
private Color(String name, int index) {
this.name = name;
this.index = index;
}
// 普通方法
public static String getName(int index) {
for (Color c : Color.values()) {
if (c.getIndex() == index) {
return c.name;
}
}
return null;
}
// get set 方法
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getIndex() {
return index;
}
public void setIndex(int index) {
this.index = index;
}
}
用法四:覆蓋枚舉的方法
下面給出一個(gè)toString()方法覆蓋的例子碘橘。
public class Test {
public enum Color {
RED("紅色", 1), GREEN("綠色", 2), BLANK("白色", 3), YELLO("黃色", 4);
// 成員變量
private String name;
private int index;
// 構(gòu)造方法
private Color(String name, int index) {
this.name = name;
this.index = index;
}
// 覆蓋方法
@Override
public String toString() {
return this.index + "_" + this.name;
}
}
public static void main(String[] args) {
System.out.println(Color.RED.toString());
}
}
用法五:實(shí)現(xiàn)接口
所有的枚舉都繼承自java.lang.Enum類互订。由于Java 不支持多繼承,所以枚舉對(duì)象不能再繼承其他類痘拆。
public interface Behaviour {
void print();
String getInfo();
}
public enum Color implements Behaviour {
RED("紅色", 1), GREEN("綠色", 2), BLANK("白色", 3), YELLO("黃色", 4);
// 成員變量
private String name;
private int index;
// 構(gòu)造方法
private Color(String name, int index) {
this.name = name;
this.index = index;
}
// 接口方法
@Override
public String getInfo() {
return this.name;
}
// 接口方法
@Override
public void print() {
System.out.println(this.index + ":" + this.name);
}
}
用法六:使用接口組織枚舉
public interface Food {
enum Coffee implements Food {
BLACK_COFFEE, DECAF_COFFEE, LATTE, CAPPUCCINO
}
enum Dessert implements Food {
FRUIT, CAKE, GELATO
}
}
用法七:關(guān)于枚舉集合的使用
java.util.EnumSet和java.util.EnumMap是兩個(gè)枚舉集合仰禽。EnumSet保證集合中的元素不重復(fù);EnumMap中的 key是enum類型,而value則可以是任意類型纺蛆。關(guān)于這個(gè)兩個(gè)集合的使用就不在這里贅述吐葵,可以參考JDK文檔
枚舉和常量定義的區(qū)別
一、 通常定義常量方法
我們通常利用public final static方法定義的代碼如下桥氏,分別用1表示紅燈温峭,3表示綠燈,2表示黃燈字支。
public class Light {
/* 紅燈 */
public final static int RED = 1;
/* 綠燈 */
public final static int GREEN = 3;
/* 黃燈 */
public final static int YELLOW = 2;
}
二诚镰、 枚舉類型定義常量方法
枚舉類型的簡(jiǎn)單定義方法如下,我們似乎沒(méi)辦法定義每個(gè)枚舉類型的值祥款。比如我們定義紅燈、綠燈和黃燈的代碼可能如下:
public enum Light {
RED, GREEN, YELLOW;
}
我們只能夠表示出紅燈月杉、綠燈和黃燈刃跛,但是具體的值我們沒(méi)辦法表示出來(lái)。別急苛萎,既然枚舉類型提供了構(gòu)造函數(shù)桨昙,我們可以通過(guò)構(gòu)造函數(shù)和覆寫toString方法來(lái)實(shí)現(xiàn)。首先給Light枚舉類型增加構(gòu)造方法腌歉,然后每個(gè)枚舉類型的值通過(guò)構(gòu)造函數(shù)傳入對(duì)應(yīng)的參數(shù)蛙酪,同時(shí)覆寫toString方法,在該方法中返回從構(gòu)造函數(shù)中傳入的參數(shù)翘盖,改造后的代碼如下:
public enum Light {
// 利用構(gòu)造函數(shù)傳參
RED(1), GREEN(3), YELLOW(2);
// 定義私有變量
private int nCode;
// 構(gòu)造函數(shù)桂塞,枚舉類型只能為私有
private Light(int _nCode) {
this.nCode = _nCode;
}
@Override
public String toString() {
return String.valueOf(this.nCode);
}
}
三、 完整示例代碼
枚舉類型的完整演示代碼如下:
public class LightTest {
// 1.定義枚舉類型
public enum Light {
// 利用構(gòu)造函數(shù)傳參
RED(1), GREEN(3), YELLOW(2);
// 定義私有變量
private int nCode;
// 構(gòu)造函數(shù)馍驯,枚舉類型只能為私有
private Light(int _nCode) {
this.nCode = _nCode;
}
@Override
public String toString() {
return String.valueOf(this.nCode);
}
}
/**
*
* @param args
*/
public static void main(String[] args) {
// 1.遍歷枚舉類型
System.out.println("演示枚舉類型的遍歷 ......");
testTraversalEnum();
// 2.演示EnumMap對(duì)象的使用
System.out.println("演示EnmuMap對(duì)象的使用和遍歷.....");
testEnumMap();
// 3.演示EnmuSet的使用
System.out.println("演示EnmuSet對(duì)象的使用和遍歷.....");
testEnumSet();
}
/**
*
* 演示枚舉類型的遍歷
*/
private static void testTraversalEnum() {
Light[] allLight = Light.values();
for (Light aLight : allLight) {
System.out.println("當(dāng)前燈name:" + aLight.name());
System.out.println("當(dāng)前燈ordinal:" + aLight.ordinal());
System.out.println("當(dāng)前燈:" + aLight);
}
}
/**
*
* 演示EnumMap的使用阁危,EnumMap跟HashMap的使用差不多玛痊,只不過(guò)key要是枚舉類型
*/
private static void testEnumMap() {
// 1.演示定義EnumMap對(duì)象,EnumMap對(duì)象的構(gòu)造函數(shù)需要參數(shù)傳入,默認(rèn)是key的類的類型
EnumMap<Light, String> currEnumMap = new EnumMap<Light, String>(
Light.class);
currEnumMap.put(Light.RED, "紅燈");
currEnumMap.put(Light.GREEN, "綠燈");
currEnumMap.put(Light.YELLOW, "黃燈");
// 2.遍歷對(duì)象
for (Light aLight : Light.values()) {
System.out.println("[key=" + aLight.name() + ",value="
+ currEnumMap.get(aLight) + "]");
}
}
/**
*
* 演示EnumSet如何使用狂打,EnumSet是一個(gè)抽象類擂煞,獲取一個(gè)類型的枚舉類型內(nèi)容<BR/>
*
* 可以使用allOf方法
*/
private static void testEnumSet() {
EnumSet<Light> currEnumSet = EnumSet.allOf(Light.class);
for (Light aLightSetElement : currEnumSet) {
System.out.println("當(dāng)前EnumSet中數(shù)據(jù)為:" + aLightSetElement);
}
}
}
執(zhí)行結(jié)果如下:
演示枚舉類型的遍歷 ......
當(dāng)前燈name:RED
當(dāng)前燈ordinal:0
當(dāng)前燈:1
當(dāng)前燈name:GREEN
當(dāng)前燈ordinal:1
當(dāng)前燈:3
當(dāng)前燈name:YELLOW
當(dāng)前燈ordinal:2
當(dāng)前燈:2
演示EnmuMap對(duì)象的使用和遍歷.....
[key=RED,value=紅燈]
[key=GREEN,value=綠燈]
[key=YELLOW,value=黃燈]
演示EnmuSet對(duì)象的使用和遍歷.....
當(dāng)前EnumSet中數(shù)據(jù)為:1
當(dāng)前EnumSet中數(shù)據(jù)為:3
當(dāng)前EnumSet中數(shù)據(jù)為:2
四、 通常定義常量方法和枚舉定義常量方法區(qū)別
-
代碼:
public class State {
public static final int ON = 1; public static final Int OFF= 0;
}
有什么不好了趴乡,大家都這樣用了很長(zhǎng)時(shí)間了对省,沒(méi)什么問(wèn)題啊。
首先晾捏,它不是類型安全的蒿涎。你必須確保是int
其次,你還要確保它的范圍是0和1
最后粟瞬,很多時(shí)候你打印出來(lái)的時(shí)候同仆,你只看到 1 和0 ,
但其沒(méi)有看到代碼的人并不知道你的企圖裙品,拋棄你所有舊的public static final常量
可以創(chuàng)建一個(gè)enum類俗批,把它看做一個(gè)普通的類。除了它不能繼承其他類了市怎。(java是單繼承岁忘,它已經(jīng)繼承了Enum),可以添加其他方法,覆蓋它本身的方法
switch()參數(shù)可以使用enum了
values()方法是編譯器插入到enum定義中的static方法区匠,所以干像,當(dāng)你將enum實(shí)例向上轉(zhuǎn)型為父類Enum是,values()就不可訪問(wèn)了驰弄。解決辦法:在Class中有一個(gè)getEnumConstants()方法麻汰,所以即便Enum接口中沒(méi)有values()方法,我們?nèi)匀豢梢酝ㄟ^(guò)Class對(duì)象取得所有的enum實(shí)例
無(wú)法從enum繼承子類戚篙,如果需要擴(kuò)展enum中的元素五鲫,在一個(gè)接口的內(nèi)部,創(chuàng)建實(shí)現(xiàn)該接口的枚舉岔擂,以此將元素進(jìn)行分組位喂。達(dá)到將枚舉元素進(jìn)行分組。
使用EnumSet代替標(biāo)志乱灵。enum要求其成員都是唯一的塑崖,但是enum中不能刪除添加元素。
EnumMap的key是enum痛倚,value是任何其他Object對(duì)象规婆。
enum允許程序員為eunm實(shí)例編寫方法。所以可以為每個(gè)enum實(shí)例賦予各自不同的行為。
使用enum的職責(zé)鏈(Chain of Responsibility) .這個(gè)關(guān)系到設(shè)計(jì)模式的職責(zé)鏈模式聋呢。以多種不同的方法來(lái)解決一個(gè)問(wèn)題苗踪。然后將他們鏈接在一起。當(dāng)一個(gè)請(qǐng)求到來(lái)時(shí)削锰,遍歷這個(gè)鏈通铲,直到鏈中的某個(gè)解決方案能夠處理該請(qǐng)求。
使用enum的狀態(tài)機(jī)
使用enum多路分發(fā)