新聞中心
枚舉是JDK1.5引入的新特性。被enum關(guān)鍵字修飾的類就是一個枚舉類。

創(chuàng)新互聯(lián)建站服務(wù)項(xiàng)目包括澤州網(wǎng)站建設(shè)、澤州網(wǎng)站制作、澤州網(wǎng)頁制作以及澤州網(wǎng)絡(luò)營銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢、行業(yè)經(jīng)驗(yàn)、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機(jī)構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,澤州網(wǎng)站推廣取得了明顯的社會效益與經(jīng)濟(jì)效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到澤州省份的部分城市,未來相信會繼續(xù)擴(kuò)大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!
關(guān)于枚舉,阿里巴巴開發(fā)手冊有這樣兩條建議:
枚舉類名帶上 Enum 后綴,枚舉成員名稱需要全大寫,單詞間用下劃線隔開。
如果變量值僅在一個固定范圍內(nèi)變化用 enum 類型來定義。
一 枚舉類有哪些特點(diǎn)
創(chuàng)建一個ColorEnum的枚舉類,通過編譯,再反編譯看看它發(fā)生了哪些變化。
- public enum ColorEnum {
- RED,GREEN,BULE;
- }
使用命令javac ColorEnum.java進(jìn)行編譯生成class文件,然后再用命令javap -p ColorEnum.class進(jìn)行反編譯。
去掉包名,反編譯后的內(nèi)容如下:
- public final class ColorEnum extends Enum{
- public static final ColorEnum GREEN;
- public static final ColorEnum BULE;
- private static final ColorEnum[] $VALUES;
- public static ColorEnum[] values();
- public static ColorEnum valueOf(java.lang.String);
- private ColorEnum();
- static {};
- }
- 枚舉類被final修飾,因此枚舉類不能被繼承;
- 枚舉類默認(rèn)繼承了Enum類,java不支持多繼承,因此枚舉類不能繼承其他類;
- 枚舉類的構(gòu)造器是private修飾的,因此其他類不能通過構(gòu)造器來獲取對象;
- 枚舉類的成員變量是static修飾的,可以用類名.變量來獲取對象;
- values()方法是獲取所有的枚舉實(shí)例;
- valueOf(java.lang.String)是根據(jù)名稱獲取對應(yīng)的實(shí)例;
二 枚舉創(chuàng)建線程安全的單例模式
- public enum SingletonEnum {
- INSTANCE;
- public void doSomething(){
- // dosomething...
- }
- }
這樣一個單例模式就創(chuàng)建好了,通過SingletonEnum.INSTANCE來獲取對象就可以了。
2.1 序列化造成單例模式不安全
一個類如果如果實(shí)現(xiàn)了序列化接口,則可能破壞單例。每次反序列化一個序列化的一個實(shí)例對象都會創(chuàng)建一個新的實(shí)例。
枚舉序列化是由JVM保證的,每一個枚舉類型和定義的枚舉變量在JVM中都是唯一的,在枚舉類型的序列化和反序列化上,Java做了特殊的規(guī)定:在序列化時Java僅僅是將枚舉對象的name屬性輸出到結(jié)果中,反序列化的時候則是通過java.lang.Enum的valueOf方法來根據(jù)名字查找枚舉對象。同時,編譯器是不允許任何對這種序列化機(jī)制的定制的并禁用了writeObject、readObject、readObjectNoData、writeReplace和readResolve等方法,從而保證了枚舉實(shí)例的唯一性。
2.2 反射造成單例模式不安全
通過反射強(qiáng)行調(diào)用私有構(gòu)造器來生成實(shí)例對象,造成單例模式不安全。
- Class> aClass = Class.forName("xx.xx.xx");
- Constructor> constructor = aClass.getDeclaredConstructor(String.class);
- SingletonEnum singleton = (SingletonEnum) constructor.newInstance("Java旅途");
但是使用枚舉創(chuàng)建的單例完全不用考慮這個問題,來看看newInstance的源碼!
- public T newInstance(Object ... initargs)
- throws InstantiationException, IllegalAccessException,
- IllegalArgumentException, InvocationTargetException
- {
- if (!override) {
- if (!Reflection.quickCheckMemberAccess(clazz, modifiers)) {
- Class> caller = Reflection.getCallerClass();
- checkAccess(caller, clazz, null, modifiers);
- }
- }
- // 如果是枚舉類型,直接拋出異常,不讓創(chuàng)建實(shí)例對象!
- if ((clazz.getModifiers() & Modifier.ENUM) != 0)
- throw new IllegalArgumentException("Cannot reflectively create enum objects");
- ConstructorAccessor ca = constructorAccessor; // read volatile
- if (ca == null) {
- ca = acquireConstructorAccessor();
- }
- @SuppressWarnings("unchecked")
- T inst = (T) ca.newInstance(initargs);
- return inst;
- }
如果是enum類型,則直接拋出異常Cannot reflectively create enum objects,無法通過反射創(chuàng)建實(shí)例對象!
三 通過枚舉消除if/else
假如要寫一套加密接口,分別給小程序、app和web端來使用,但是這三種客戶端的加密方式不一樣。一般情況下我們會傳一個類型type來判斷來源,然后調(diào)用對應(yīng)的解密方法即可。代碼如下:
- if("WEIXIN".equals(type)){
- // dosomething
- }else if("APP".equals(type)){
- // dosomething
- }else if("WEB".equals(type)){
- // dosomething
- }
現(xiàn)在使用枚舉來消除這些if/else。
寫一個加密用的接口,有加密和解密兩個方法。然后用不同的算法去實(shí)現(xiàn)這個接口完成加解密。
- public interface Util {
- // 解密
- String decrypt();
- // 加密
- String encrypt();
- }
創(chuàng)建一個枚舉類來實(shí)現(xiàn)這個接口
- public enum UtilEnum implements Util {
- WEIXIN {
- @Override
- public String decrypt() {
- return "微信解密";
- }
- @Override
- public String encrypt() {
- return "微信加密";
- }
- },
- APP {
- @Override
- public String decrypt() {
- return "app解密";
- }
- @Override
- public String encrypt() {
- return "app加密";
- }
- },
- WEB {
- @Override
- public String decrypt() {
- return "web解密";
- }
- @Override
- public String encrypt() {
- return "web加密";
- }
- };
- }
最后,獲取到type后,直接調(diào)用解密方法就行了。
- String decryptMessage = UtilEnum.valueOf(type).decrypt();
以后,如果新增了一個其他加密方式,只需要修改上面的枚舉類就完成了,業(yè)務(wù)代碼都不需要改動。
這就是枚舉類比較高級的兩個用法。
本文轉(zhuǎn)載自微信公眾號「Java旅途」,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請聯(lián)系Java旅途公眾號。
分享名稱:Coder,我懷疑你并不會枚舉
地址分享:http://www.dlmjj.cn/article/ccddioh.html


咨詢
建站咨詢
