Java枚舉與.net枚舉區(qū)別詳解
通過(guò)一段時(shí)間的項(xiàng)目實(shí)踐,發(fā)現(xiàn)java中的枚舉與.net中的枚舉有很大的差別,初期造成了我對(duì)java中的枚舉一些錯(cuò)誤理解及部分有缺陷的應(yīng)用,其實(shí)追其原因還是因?yàn)槲視?huì)習(xí)慣性的認(rèn)為java的枚舉在作用以及定義上與.net應(yīng)該是差不多的,畢竟兩者都是高級(jí)語(yǔ)言,語(yǔ)言上也有很多相似之處。這就是老師傅常說(shuō)的新手好教,老兵不好教的原因,新手腦子一片空白不會(huì)有任何干擾,老兵總會(huì)以自己曾經(jīng)的某些經(jīng)驗(yàn)與新知識(shí)做對(duì)比。
習(xí)慣性觀點(diǎn)一:枚舉的定義應(yīng)該與.net相同,比如在.net中我們可以這樣定義枚舉。
public enum EItemDataType { Real=1, Service=2 }
但java中并不能如此瀟灑的書(shū)寫(xiě)枚舉,可能需要類(lèi)似這樣寫(xiě):
public enum EItemDataType { Real(1),Service(2); private int value; private EItemDataType(int value) { this.value = value; } public int getValue() { return value; } public static EItemDataType valueOf(int value) { switch (value) { case 1: return EItemDataType.Real; case 2: return EItemDataType.Service; default: return null; } } }
發(fā)現(xiàn).net要比java簡(jiǎn)單的多,注意幾個(gè)方法:
valueOf的方法:看作用是為了根據(jù)一個(gè)枚舉的數(shù)值來(lái)得到枚舉,這個(gè)功能很常見(jiàn),但在.net中就不需要這樣麻煩了,可以直接將數(shù)據(jù)強(qiáng)轉(zhuǎn)成枚舉,比如:
var itemType=(EItemDataType)1;
getValue的方式,明顯是需要將一個(gè)枚舉轉(zhuǎn)換成它所對(duì)應(yīng)的值,.net中也不需要調(diào)用方法來(lái)取值,也可以強(qiáng)轉(zhuǎn),比如:
var itemTypeValue=(int)EItemDataType.Real;
私有構(gòu)造函數(shù),我們可以傳多少參數(shù),比如常見(jiàn)的我們需要顯示這個(gè)枚舉值對(duì)應(yīng)的中文描述,在java中我們只需要在構(gòu)造函數(shù)中增加一個(gè)name參數(shù)就可以了,但在.net中因?yàn)闆](méi)有這貨不能這樣做,但可以通過(guò) Atrribute來(lái)完成。
public enum EItemDataType { [Description('實(shí)物')] Real=1, [Description('服務(wù)')] Service=2 }
習(xí)慣性觀點(diǎn)二:因?yàn)?net的枚舉是個(gè)值類(lèi)型,所以我理所當(dāng)然的會(huì)認(rèn)為java的枚舉也是一個(gè)值類(lèi)型。之前對(duì).net的理解就是將一些數(shù)值以更加可讀性的方式體現(xiàn)在程序中,比如訂單狀態(tài),訂單類(lèi)型等等,比如:
//枚舉值可讀性更強(qiáng)if(orderInfo.orderStatus.equals(EOrderStatus.Shipped)){ //do something}//一般不這樣寫(xiě),0可讀性不強(qiáng)if(orderInfo.orderStatus==0){ //do something}
枚舉類(lèi)型的自說(shuō)明:
編譯后的文件中找到了EItemDataType.class這個(gè)文件,這說(shuō)明java的枚舉其實(shí)和普通的類(lèi)是一樣的,既然是一個(gè)類(lèi),那么肯定不是值類(lèi)型了,下圖中的引用類(lèi)型中包含class type。
編譯之后所對(duì)應(yīng)的字節(jié)碼到底是什么樣的:
public final class EItemDataType extends java.lang.Enum<EItemDataType> { public static final EItemDataType Real; public static final EItemDataType Service; static {}; Code: 0: new #1 // class EItemDataType 3: dup 4: ldc #15 // String Real 6: iconst_0 7: iconst_1 8: invokespecial #16 // Method '<init>':(Ljava/lang/String;II)V 11: putstatic #20 // Field Real:LEItemDataType; 14: new #1 // class EItemDataType 17: dup 18: ldc #22 // String Service 20: iconst_1 21: iconst_2 22: invokespecial #16 // Method '<init>':(Ljava/lang/String;II)V 25: putstatic #23 // Field Service:LEItemDataType; 28: iconst_2 29: anewarray #1 // class EItemDataType 32: dup 33: iconst_0 34: getstatic #20 // Field Real:LEItemDataType; 37: aastore 38: dup 39: iconst_1 40: getstatic #23 // Field Service:LEItemDataType; 43: aastore 44: putstatic #25 // Field ENUM$VALUES:[LEItemDataType; 47: return public int getValue(); Code: 0: aload_0 1: getfield #32 // Field value:I 4: ireturn public static EItemDataType valueOf(int); Code: 0: iload_0 1: tableswitch { // 1 to 2 1: 24 2: 28default: 32 } 24: getstatic #20 // Field Real:LEItemDataType; 27: areturn 28: getstatic #23 // Field Service:LEItemDataType; 31: areturn 32: aconst_null 33: areturn public static EItemDataType[] values(); Code: 0: getstatic #25 // Field ENUM$VALUES:[LEItemDataType; 3: dup 4: astore_0 5: iconst_0 6: aload_0 7: arraylength 8: dup 9: istore_1 10: anewarray #1 // class EItemDataType 13: dup 14: astore_2 15: iconst_0 16: iload_1 17: invokestatic #42 // Method java/lang/System.arraycopy:(Ljava/lang/Object;ILjava/lang/Object;II)V 20: aload_2 21: areturn public static EItemDataType valueOf(java.lang.String); Code: 0: ldc #1 // class EItemDataType 2: aload_0 3: invokestatic #49 // Method java/lang/Enum.valueOf:(Ljava/lang/Class;Ljava/lang/String;)Ljava/lang/Enum; 6: checkcast #1 // class EItemDataType 9: areturn}
是個(gè)final類(lèi)型的,不允許繼承自其它類(lèi)型
繼承了java.lang.Enum類(lèi),更說(shuō)明這個(gè)枚舉就是個(gè)class
public final class EItemDataType extends java.lang.Enum<EItemDataType> {
所有的枚舉值都被定義成靜態(tài)值了,且以常量形式存在
public static final EItemDataType Real;
再看下一個(gè)特殊的方法,由于枚舉繼承了java.lang.Enum這個(gè)類(lèi),那么它自然擁有一些實(shí)用的方法:
public static EItemDataType valueOf(java.lang.String);
這是個(gè)字符串參數(shù)類(lèi)型的方法,和我上面定義的valueOf(int value)很像,其目的都是根據(jù)一定的條件獲取枚舉值,只不過(guò)方式不同而已,前者是自帶的根據(jù)枚舉值toString的結(jié)果來(lái)反向獲取枚舉值,與toString的對(duì)應(yīng),比如:EItemDataType.Real.toString()它等于“Real”,再調(diào)用EItemDataType.valueOf('Reail'),它等于EItemDataType.Real這個(gè)值。自定義的valueOf(int value)方式個(gè)人感覺(jué)并不太好,因?yàn)槿菀着c自帶的那個(gè)方法沖突,最好是改個(gè)名稱(chēng),比如value什么。
最后我們?cè)賮?lái)看下枚舉所能實(shí)現(xiàn)的奇葩功能:?jiǎn)卫?之前學(xué)習(xí).net時(shí)寫(xiě)的日記:老生常談:單件模式)。剛開(kāi)始看到j(luò)ava的單例可以通過(guò)枚舉實(shí)現(xiàn)時(shí),我都驚呆了,最大的反應(yīng)是枚舉是個(gè)存儲(chǔ)值的怎么和單例有關(guān)系?單例不是class的事嗎?其實(shí)通過(guò)上面的理解,枚舉就是個(gè)類(lèi),那么再想想單例就不會(huì)有什么疑問(wèn)了,把它當(dāng)成一個(gè)普通類(lèi)不就好了,我們看一個(gè)簡(jiǎn)單的計(jì)數(shù)的例子:按照上面字節(jié)碼的結(jié)構(gòu),這個(gè)INSTANCE2會(huì)被定義成一個(gè)靜態(tài)變量,正是利用靜態(tài)變量唯一性的特性來(lái)實(shí)現(xiàn)了單例,而且是線程安全的。
public enum SafeSingleton implements Serializable { INSTANCE2; int count; public void addCount(int i) { this.count+=i; } public void printCount() { System.out.println(this.count); }}
下面這段程序會(huì)輸出5050
for(int i=1;i<=100;i++){ SafeSingleton.INSTANCE2.addCount(i); } SafeSingleton.INSTANCE2.printCount();
總結(jié)
java中的枚舉是一個(gè)比較特殊的數(shù)據(jù)類(lèi)型,除了具備值存儲(chǔ)的能力還擁有class特性,作用范圍相比.net要大,但實(shí)現(xiàn)更加復(fù)雜些。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
1. CSS3實(shí)現(xiàn)動(dòng)態(tài)翻牌效果 仿百度貼吧3D翻牌一次動(dòng)畫(huà)特效2. 三個(gè)不常見(jiàn)的 HTML5 實(shí)用新特性簡(jiǎn)介3. asp批量添加修改刪除操作示例代碼4. msxml3.dll 錯(cuò)誤 800c0019 系統(tǒng)錯(cuò)誤:-2146697191解決方法5. 刪除docker里建立容器的操作方法6. asp在iis7報(bào)錯(cuò)行號(hào)不準(zhǔn)問(wèn)題的解決方法7. 推薦一個(gè)好看Table表格的css樣式代碼詳解8. ASP中解決“對(duì)象關(guān)閉時(shí),不允許操作。”的詭異問(wèn)題……9. asp讀取xml文件和記數(shù)10. 概述IE和SQL2k開(kāi)發(fā)一個(gè)XML聊天程序
