java枚舉類型enum的使用
最近跟同事討論問題的時(shí)候,突然同事提到我們?yōu)槭裁磈ava中定義的常量值不采用enmu枚舉類型,而采用public final static 類型來定義呢?以前我們都是采用這種方式定義的,很少采用enum定義,所以也都沒有注意過,面對(duì)突入起來的問題,還真有點(diǎn)不太清楚為什么有這樣的定義。既然不明白就抽時(shí)間研究下吧。
Java中的枚舉類型采用關(guān)鍵字enum來定義,從jdk1.5才有的新類型,所有的枚舉類型都是繼承自Enum類型。要了解枚舉類型,建議大家先打開jdk中的Enum類簡單讀一下,這個(gè)類里面定義了很多PRotected方法,比如構(gòu)造函數(shù),如果要使用這些方法我們可以把枚舉類型定義到當(dāng)前類中。每個(gè)枚舉類型,都有自己的名字和順序,當(dāng)我們輸出一個(gè)枚舉類型的時(shí)候,會(huì)輸入枚舉類型的name,具體可以參考下面的例子。
一、 通常定義常量方法
我們通常利用public final static方法定義的代碼如下,分別用1表示紅燈,3表示綠燈,2表示黃燈。
package com.csdn.myEnum; publicclass Light { /*紅燈*/ publicfinalstaticintRED=1; /*綠燈*/ publicfinalstaticintGREEN=3; /*黃燈*/ publicfinalstaticintYELLOW=2; } |
二、 枚舉類型定義常量方法
枚舉類型的簡單定義方法如下,我們似乎沒辦法定義每個(gè)枚舉類型的值。比如我們定義紅燈、綠燈和黃燈的代碼可能如下:
publicenum Light { RED,GREEN,YELLOW; } |
我們只能夠表示出紅燈、綠燈和黃燈,但是具體的值我們沒辦法表示出來。別急,既然枚舉類型提供了構(gòu)造函數(shù),我們可以通過構(gòu)造函數(shù)和覆寫toString方法來實(shí)現(xiàn)。首先給Light枚舉類型增加構(gòu)造方法,然后每個(gè)枚舉類型的值通過構(gòu)造函數(shù)傳入對(duì)應(yīng)的參數(shù),同時(shí)覆寫toString方法,在該方法中返回從構(gòu)造函數(shù)中傳入的參數(shù),改造后的代碼如下:
publicenum Light { //利用構(gòu)造函數(shù)傳參 RED(1),GREEN(3),YELLOW(2);
//定義私有變量 privateintnCode;
//構(gòu)造函數(shù),枚舉類型只能為私有 private Light(int _nCode) { this.nCode = _nCode; }
@Override public String toString() { return String.valueOf(this.nCode); } } |
三、 完整示例代碼
枚舉類型的完整演示代碼如下:
package com.csdn.myEnum;
import java.util.EnumMap; import java.util.EnumSet;
publicclass LightTest {
// 1.定義枚舉類型 publicenum Light { //利用構(gòu)造函數(shù)傳參 RED(1),GREEN(3),YELLOW(2);
//定義私有變量 privateintnCode;
//構(gòu)造函數(shù),枚舉類型只能為私有 private Light(int _nCode) { this.nCode = _nCode; }
@Override public String toString() { return String.valueOf(this.nCode); } }
/** *@paramargs */ publicstaticvoid 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(); }
/** *演示枚舉類型的遍歷 */ privatestaticvoid 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的使用差不多,只不過key要是枚舉類型 */ privatestaticvoid 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方法 */ privatestaticvoid 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ū)別
以下內(nèi)容可能有些無聊,但絕對(duì)值得一窺
1. 代碼:
public class State {
public static final int ON = 1;
public static final Int OFF= 0;
}
有什么不好了,大家都這樣用了很長時(shí)間了,沒什么問題啊。
首先,它不是類型安全的。你必須確保是int
其次,你還要確保它的范圍是0和1
最后,很多時(shí)候你打印出來的時(shí)候,你只看到 1 和0 ,
但其沒有看到代碼的人并不知道你的企圖,拋棄你所有舊的public static final常量吧
2. 可以創(chuàng)建一個(gè)enum類,把它看做一個(gè)普通的類。除了它不能繼承其他類了。(java是單繼承,它已經(jīng)繼承了Enum),
可以添加其他方法,覆蓋它本身的方法
3. switch()參數(shù)可以使用enum了
4. values()方法是編譯器插入到enum定義中的static方法,所以,當(dāng)你將enum實(shí)例向上轉(zhuǎn)型為父類Enum是,values()就不可訪問了。解決辦法:在Class中有一個(gè)getEnumConstants()方法,所以即便Enum接口中沒有values()方法,我們?nèi)匀豢梢酝ㄟ^Class對(duì)象取得所有的enum實(shí)例
5. 無法從enum繼承子類,如果需要擴(kuò)展enum中的元素,在一個(gè)接口的內(nèi)部,創(chuàng)建實(shí)現(xiàn)該接口的枚舉,以此將元素進(jìn)行分組。達(dá)到將枚舉元素進(jìn)行分組。
6. 使用EnumSet代替標(biāo)志。enum要求其成員都是唯一的,但是enum中不能刪除添加元素。
7. EnumMap的key是enum,value是任何其他Object對(duì)象。
8. enum允許程序員為eunm實(shí)例編寫方法。所以可以為每個(gè)enum實(shí)例賦予各自不同的行為。
9. 使用enum的職責(zé)鏈(Chain of Responsibility) .這個(gè)關(guān)系到設(shè)計(jì)模式的職責(zé)鏈模式。以多種不同的方法來解決一個(gè)問題。然后將他們鏈接在一起。當(dāng)一個(gè)請(qǐng)求到來時(shí),遍歷這個(gè)鏈,直到鏈中的某個(gè)解決方案能夠處理該請(qǐng)求。
10. 使用enum的狀態(tài)機(jī)
11. 使用enum多路分發(fā)
原文轉(zhuǎn)自于: http://blog.csdn.net/wgw335363240/article/details/6359614
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注