Java枚舉與.net枚舉區(qū)別詳解
通過一段時(shí)間的項(xiàng)目實(shí)踐,發(fā)現(xiàn)java中的枚舉與.net中的枚舉有很大的差別,初期造成了我對java中的枚舉一些錯(cuò)誤理解及部分有缺陷的應(yīng)用,其實(shí)追其原因還是因?yàn)槲視?huì)習(xí)慣性的認(rèn)為java的枚舉在作用以及定義上與.net應(yīng)該是差不多的,畢竟兩者都是高級(jí)語言,語言上也有很多相似之處。這就是老師傅常說的新手好教,老兵不好教的原因,新手腦子一片空白不會(huì)有任何干擾,老兵總會(huì)以自己曾經(jīng)的某些經(jīng)驗(yàn)與新知識(shí)做對比。
習(xí)慣性觀點(diǎn)一:枚舉的定義應(yīng)該與.net相同,比如在.net中我們可以這樣定義枚舉。
public enum EItemDataType
{
Real=1,
Service=2
}
但java中并不能如此瀟灑的書寫枚舉,可能需要類似這樣寫:
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簡單的多,注意幾個(gè)方法:
valueOf的方法:看作用是為了根據(jù)一個(gè)枚舉的數(shù)值來得到枚舉,這個(gè)功能很常見,但在.net中就不需要這樣麻煩了,可以直接將數(shù)據(jù)強(qiáng)轉(zhuǎn)成枚舉,比如:
var itemType=(EItemDataType)1;
getValue的方式,明顯是需要將一個(gè)枚舉轉(zhuǎn)換成它所對應(yīng)的值,.net中也不需要調(diào)用方法來取值,也可以強(qiáng)轉(zhuǎn),比如:
var itemTypeValue=(int)EItemDataType.Real;
私有構(gòu)造函數(shù),我們可以傳多少參數(shù),比如常見的我們需要顯示這個(gè)枚舉值對應(yīng)的中文描述,在java中我們只需要在構(gòu)造函數(shù)中增加一個(gè)name參數(shù)就可以了,但在.net中因?yàn)闆]有這貨不能這樣做,但可以通過 Atrribute來完成。
public enum EItemDataType
{
[Description("實(shí)物")]
Real=1,
[Description("服務(wù)")]
Service=2
}
習(xí)慣性觀點(diǎn)二:因?yàn)?net的枚舉是個(gè)值類型,所以我理所當(dāng)然的會(huì)認(rèn)為java的枚舉也是一個(gè)值類型。之前對.net的理解就是將一些數(shù)值以更加可讀性的方式體現(xiàn)在程序中,比如訂單狀態(tài),訂單類型等等,比如:
//枚舉值可讀性更強(qiáng)
if(orderInfo.orderStatus.equals(EOrderStatus.Shipped)){
//do something
}
//一般不這樣寫,0可讀性不強(qiáng)
if(orderInfo.orderStatus==0){
//do something
}
枚舉類型的自說明:
編譯后的文件中找到了EItemDataType.class這個(gè)文件,這說明java的枚舉其實(shí)和普通的類是一樣的,既然是一個(gè)類,那么肯定不是值類型了,下圖中的引用類型中包含class type。
編譯之后所對應(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: 28
default: 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類型的,不允許繼承自其它類型
繼承了java.lang.Enum類,更說明這個(gè)枚舉就是個(gè)class
public final class EItemDataType extends java.lang.Enum<EItemDataType> {
所有的枚舉值都被定義成靜態(tài)值了,且以常量形式存在
public static final EItemDataType Real;
再看下一個(gè)特殊的方法,由于枚舉繼承了java.lang.Enum這個(gè)類,那么它自然擁有一些實(shí)用的方法:
public static EItemDataType valueOf(java.lang.String);
這是個(gè)字符串參數(shù)類型的方法,和我上面定義的valueOf(int value)很像,其目的都是根據(jù)一定的條件獲取枚舉值,只不過方式不同而已,前者是自帶的根據(jù)枚舉值toString的結(jié)果來反向獲取枚舉值,與toString的對應(yīng),比如:EItemDataType.Real.toString()它等于“Real”,再調(diào)用EItemDataType.valueOf("Reail"),它等于EItemDataType.Real這個(gè)值。自定義的valueOf(int value)方式個(gè)人感覺并不太好,因?yàn)槿菀着c自帶的那個(gè)方法沖突,最好是改個(gè)名稱,比如value什么。
最后我們再來看下枚舉所能實(shí)現(xiàn)的奇葩功能:單例(之前學(xué)習(xí).net時(shí)寫的日記:老生常談:單件模式)。剛開始看到j(luò)ava的單例可以通過枚舉實(shí)現(xiàn)時(shí),我都驚呆了,最大的反應(yīng)是枚舉是個(gè)存儲(chǔ)值的怎么和單例有關(guān)系?單例不是class的事嗎?其實(shí)通過上面的理解,枚舉就是個(gè)類,那么再想想單例就不會(huì)有什么疑問了,把它當(dāng)成一個(gè)普通類不就好了,我們看一個(gè)簡單的計(jì)數(shù)的例子:按照上面字節(jié)碼的結(jié)構(gòu),這個(gè)INSTANCE2會(huì)被定義成一個(gè)靜態(tài)變量,正是利用靜態(tà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ù)類型,除了具備值存儲(chǔ)的能力還擁有class特性,作用范圍相比.net要大,但實(shí)現(xiàn)更加復(fù)雜些。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
解決springboot3.2集成mybatis-plus3.5.4.1報(bào)錯(cuò)的問題
這篇文章給大家介紹了如何解決springboot3.2集成mybatis-plus3.5.4.1報(bào)錯(cuò)的問題,文中通過圖文介紹的非常詳細(xì),具有一定的參考價(jià)值,需要的朋友可以參考下2023-12-12
SpringBoot啟動(dòng)應(yīng)用及回調(diào)監(jiān)聽原理解析
這篇文章主要介紹了SpringBoot啟動(dòng)應(yīng)用及回調(diào)監(jiān)聽原理解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12
Spring中的spring-retry重試機(jī)制解析
這篇文章主要介紹了Spring中的spring-retry重試機(jī)制解析,spring-retry可以通過注解,在不入侵原有業(yè)務(wù)邏輯代碼的方式下,優(yōu)雅的實(shí)現(xiàn)重處理功能,在spring-retry中,所有配置都是基于簡單注釋的,需要的朋友可以參考下2024-01-01
Java中checkbox實(shí)現(xiàn)跨頁多選的方法
最近做了一個(gè)項(xiàng)目其中遇到這樣的需求,要實(shí)現(xiàn)checkbox跨頁多選功能,經(jīng)過小編整理,順利解決,今天小編給大家分享Java中checkbox實(shí)現(xiàn)跨頁多選的方法,需要的的朋友參考下2017-01-01
MyBatis-Plus+Druid配置及應(yīng)用詳解
這篇文章主要介紹了MyBatis-Plus+Druid配置及應(yīng)用詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11
SpringBoot統(tǒng)一api返回風(fēng)格的實(shí)現(xiàn)
這篇文章主要介紹了SpringBoot統(tǒng)一api返回風(fēng)格的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03
Java使用Ajax實(shí)現(xiàn)跨域上傳圖片功能
這篇文章主要介紹了Java使用Ajax實(shí)現(xiàn)跨域上傳圖片功能,需要的朋友可以參考下2017-09-09
idea打開項(xiàng)目后無法顯示目錄結(jié)構(gòu),只能顯示.iml文件問題
這篇文章主要介紹了idea打開項(xiàng)目后無法顯示目錄結(jié)構(gòu),只能顯示.iml文件問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-08-08

