您好,登錄后才能下訂單哦!
小編給大家分享一下java使用枚舉的原因,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!
java為什么要使用枚舉?
假設(shè)現(xiàn)在有兩種訂單類型:預(yù)訂訂單和非預(yù)訂訂單。
需求一: 方法submitOrder根據(jù)不同訂單類型進(jìn)行不同的處理。
需求二: 給一個(gè)對(duì)象: OrderResult設(shè)置訂單類型。
下面分別用兩種枚舉方式來(lái)實(shí)現(xiàn)。
int枚舉
class IntEnumExample{ private static final PRE_ORDER = 1; private static final NOT_PRE_ORDER = 2; public void submitOrder(int orderType, OrderResult orderResult){ orderResult.setType(orderType); if(orderType == PRE_ORDER){ do something to process preOrder }else if(orderType == NOT_PRE_ORDER){ do something to process other order } } public static void main(String [] args){ IntEnumExample example = new IntEnumExample(); //passing wrong type to the method, however, no compile error and runtime exception here, the bug is hard to be discerned. example.submitOrder(3, orderResult); } }
從上面的例子可以看到,使用int枚舉有幾個(gè)缺點(diǎn):
1.int枚舉不做類型檢查,可以給上面的submitOrder方法傳入任意int值
2.代碼可讀性低,如果沒(méi)有文檔或者源碼,不知道給submitOrder傳遞的值的意義。
不僅如此,就像我之前修改項(xiàng)目中sonar掃描代碼發(fā)現(xiàn)的問(wèn)題那樣,很多人在使用int枚舉時(shí),并沒(méi)有像上例中將int值定義成static final。如果忘記定義變量為final則int枚舉的值就可以被修改,如果忘記定義變量為static,就可能出現(xiàn)使用這個(gè)int值時(shí),該int值還沒(méi)有被初始化好。
總之,會(huì)引起bug。
下面來(lái)看看使用enum如何做同樣的事兒。
class IntEnumExample{ private enum ORDER_TYPE { NOT_PRE_ORDER(1),PRE_ORDER(2); private final int value; private ORDER_TYPE(int value){ this.value = value; } } public void submitOrder(ORDER_TYPE orderType, OrderResult orderResult){ orderResult.setType(orderType); if(orderType == ORDER_TYPE.PRE_ORDER){ do something to process preOrder }else if(orderType == ORDER_TYPE.NOT_PRE_ORDER){ do something to process other order } } public static void main(String [] args){ IntEnumExample example = new IntEnumExample(); //compiler will complain error here, if argument is not the type in the enum. example.submitOrder(ORDER_TYPE.PRE_ORDER, orderResult); } }
通過(guò)上述代碼可以看到,enum很優(yōu)雅的解決了上一個(gè)例子中的問(wèn)題。
1.編譯器將對(duì)enum進(jìn)行類型檢查,類型不符合的編譯器會(huì)直接報(bào)錯(cuò)。
2.相比與int枚舉型直接傳int數(shù)值的方式,enum傳遞enum類型對(duì)象的方式,代碼可讀性更高,傳遞的枚舉類型一目了然。
3.enum類型中的對(duì)象本身就是static final的。
重要提示:
還有一點(diǎn)值得一提的是,如果有時(shí)想給每一個(gè)枚舉類型賦予一個(gè)int值,要使用上例中enum定義的方式。
private enum ORDER_TYPE { NOT_PRE_ORDER(1),PRE_ORDER(2); private final int value; private ORDER_TYPE(int value){ this.value = value; } }
enum本質(zhì)也是一個(gè)類,所以方法ORDER_TYPE(int value)是這個(gè)枚舉類型的構(gòu)造函數(shù),故每個(gè)枚舉類型在初始化的時(shí)候需要給構(gòu)造函數(shù)傳遞響應(yīng)的值,如: PRE_ORDER(2)。
這種情況下,想得到枚舉類型對(duì)應(yīng)的int數(shù)值時(shí)可以通過(guò)ORDER_TYPE.PRE_ORDER.value獲取。
單獨(dú)提下enum的這種用法是因?yàn)?,有些人可能?huì)直接使用enum中的ordinal()方法直接實(shí)現(xiàn)enum類型與int類型的關(guān)聯(lián)。ordinal()方法返回的是enum類型在被定義時(shí)的序數(shù),如ORDER_TYPE.PRE_ORDER.value.ordinal()返回值為0。所以獲取枚舉類型對(duì)應(yīng)的int數(shù)值貌似也可以通過(guò)ORDER_TYPE.PRE_ORDER.value.ordinal()+1實(shí)現(xiàn)。
不要使用ordinal()方法!不要使用ordinal()方法!不要使用ordinal()方法!重要的事情說(shuō)三遍,為什么?
序數(shù)是很不可靠的東西,序數(shù)是可以改變的,假設(shè)有一天ORDER_TYPE的定義變了,需要增加幾種類型,或者不小心換了NOT_PRE_ORDER和PRE_ORDER定義時(shí)的順序,如:
private enum ORDER_TYPE { PRE_ORDER,NOT_PRE_ORDER; }
這時(shí)就會(huì)造成很嚴(yán)重的bug,而且不好發(fā)現(xiàn),編譯時(shí),運(yùn)行時(shí)都不會(huì)有報(bào)錯(cuò)。
所以,不要依賴ordianl()方法。
以上是java使用枚舉的原因的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對(duì)大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。