Android之使用枚舉利弊及替代方案
不使用枚舉enum引起的類型不安全問題; 使用枚舉enum的利弊; 利用@interface+@IntDef/@StringDef替代enum; 新名詞記錄{typeof,@IntDef/@StringDef,@interface}
Enum 是 Java 中一種包含固定常數的類型,當我們需要預先定義一些值時,我們使用 Enum,這樣做通常為了在編譯時期避免接受額外常量引起的錯誤。
而且,Enum 增加了APK 的大小,比常量多5到10倍的內存占用,這是關于應用性能的最佳實踐.
舉一個例子,例如我們要為一個bean賦值一個person的性別屬性,因為性別只有男女,所以我們通常的做法是定義兩個整型int,來區分“男女”性別。
publicclassSexTest{privatefinalintMAN =101, WOMEN =102;privateintsex;//設置性別publicvoidsetSex(intsex){this.sex = sex;? ? }//獲取性別publicStringgetSex(){if(MAN == sex)return"男";if(WOMEN == sex)return"女";return"未知";? ? }publicstaticvoidmain(String[] args){? ? ? ? setSex(101);intsex = getSex();? ? ? ? System.out.println("sex: "+ sex);//輸出:sex: 男//設置為102入參setSex(102);? ? ? ? String resultSex = getSex();? ? ? ? System.out.println("resultSex: "+ resultSex);//輸出:resultSex: 未知}}
由上面的例子可以看出,當我們定義了一個男女的final整型作為入參時,不一定保證入參的都是我們想要的入參,這里就有一個“類型不安全”的問題出現。而枚舉就可以解決這個問題。
例如,首先定義一個枚舉類,里面有男,女兩個枚舉常量。
publicclassSexTest{publicstaticenumSex {? ? ? MAN, WOMEN? ? }privateSex sex;//設置性別publicvoidsetSex(Sex sex){this.sex = sex;? ? }//獲取性別publicStringgetSex(){if(Sex.MAN == sex)return"男";if(Sex.WOMEN == sex)return"女";return"未知";? ? }publicstaticvoidmain(String[] args){//這里的入參必須為Sex枚舉類中的其中一個枚舉常量//絕對不允許輸入沒有再Sex枚舉里面定義的常量setSex(Sex.MAN);? ? ? ? String resultSex = getSex();? ? ? ? System.out.println("resultSex: "+ resultSex);//輸出:resultSex: 男}}
所以我們可以看到,我們利用枚舉,在setSex()方法里面對入參做了枚舉Sex的限制,對于我們想輸入任何非枚舉類Sex里面定義的枚舉常量,編譯都是不能通過的。這就很好的限制了入參混亂的問題。
但是,“every coin has two sides”。馬克思告訴我們,一切都是矛盾體,事物必然有好的一面也有不好的一面。對于枚舉同樣適用,下面我們來看下適用枚舉的缺點。
每一個枚舉值都是一個對象,在使用它時會增加額外的內存消耗,所以枚舉相比與 Integer 和 String 會占用更多的內存。
較多的使用 Enum 會增加 DEX 文件的大小,會造成運行時更多的開銷,使我們的應用需要更多的空間。
如果你的應用使用很多的 Enum ,最好使用Integer 或 String 替代他們,但是這樣還會有問題。
既然都說到這個份上了,那么有什么比較好的解決方法呢?
既然是因為參數的類型太泛了造成的類型不安全,那么我只要將參數限定在某一個類型集合里面,不就大功告成了?!
是滴,一下就是要將的@IntDef/@StringDef + @interface來進行限定參數。
首先你要在build.gradle文件中添加依賴
dependencies { compile ‘com.android.support:support-annotations:24.2.0’ }
然后再使用,代碼如下: public class SexTest { private final int MAN = 101, WOMEN = 102;
@IntDef({MAN, WOMEN})//限定為MAN,WOMEN@Retention(RetentionPolicy.SOURCE)//表示注解所存活的時間,在運行時,而不會存在. class 文件.public@interfaceSex {//接口,定義新的注解類型}publicvoidsetSex(@Sexintsex){this.sex = sex;publicstaticvoidmain(String[] args){? ? ? ? setSex(MAN);? ? }}
如果我們嘗試在調用setSex()方法的時候,傳入不在限定之內的值,那么編譯就不會通過,有錯誤提示。同理,我們也可以使用@StringDef。
可以看到,如果不適用枚舉,將會帶來類型不安全的問題。一般情況下,我們在很多地方都會使用到枚舉,因為方便和簡潔。但是使用枚舉也會產生占用內存過高等情況。所以我們可以有了自定義的方案,來限定我們使用的類型范圍。
以上就是全部內容,如有任何問題,請及時與我聯系,謝謝!