亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java枚舉類的使用方法是什么

發布時間:2021-10-29 09:43:40 來源:億速云 閱讀:123 作者:iii 欄目:編程語言

本篇內容介紹了“Java枚舉類的使用方法是什么”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!

一 枚舉類有哪些特點

創建一個ColorEnum的枚舉類,通過編譯,再反編譯看看它發生了哪些變化。

public enum ColorEnum {     RED,GREEN,BULE; }

使用命令javac ColorEnum.java進行編譯生成class文件,然后再用命令javap -p  ColorEnum.class進行反編譯。

Java枚舉類的使用方法是什么

去掉包名,反編譯后的內容如下:

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 {}; }
  1. 枚舉類被final修飾,因此枚舉類不能被繼承;

  2. 枚舉類默認繼承了Enum類,java不支持多繼承,因此枚舉類不能繼承其他類;

  3. 枚舉類的構造器是private修飾的,因此其他類不能通過構造器來獲取對象;

  4. 枚舉類的成員變量是static修飾的,可以用類名.變量來獲取對象;

  5. values()方法是獲取所有的枚舉實例;

  6. valueOf(java.lang.String)是根據名稱獲取對應的實例;

二 枚舉創建線程安全的單例模式

public enum  SingletonEnum {      INSTANCE;      public void doSomething(){         // dosomething...     } }

這樣一個單例模式就創建好了,通過SingletonEnum.INSTANCE來獲取對象就可以了。

2.1 序列化造成單例模式不安全

一個類如果如果實現了序列化接口,則可能破壞單例。每次反序列化一個序列化的一個實例對象都會創建一個新的實例。

枚舉序列化是由JVM保證的,每一個枚舉類型和定義的枚舉變量在JVM中都是唯一的,在枚舉類型的序列化和反序列化上,Java做了特殊的規定:在序列化時Java僅僅是將枚舉對象的name屬性輸出到結果中,反序列化的時候則是通過java.lang.Enum的valueOf方法來根據名字查找枚舉對象。同時,編譯器是不允許任何對這種序列化機制的定制的并禁用了writeObject、readObject、readObjectNoData、writeReplace和readResolve等方法,從而保證了枚舉實例的唯一性。

2.2 反射造成單例模式不安全

通過反射強行調用私有構造器來生成實例對象,造成單例模式不安全。

Class<?> aClass = Class.forName("xx.xx.xx"); Constructor<?> constructor = aClass.getDeclaredConstructor(String.class); SingletonEnum singleton = (SingletonEnum) constructor.newInstance("Java旅途");

但是使用枚舉創建的單例完全不用考慮這個問題,來看看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);         }     }     // 如果是枚舉類型,直接拋出異常,不讓創建實例對象!     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,無法通過反射創建實例對象!

三 通過枚舉消除if/else

假如要寫一套加密接口,分別給小程序、app和web端來使用,但是這三種客戶端的加密方式不一樣。一般情況下我們會傳一個類型type來判斷來源,然后調用對應的解密方法即可。代碼如下:

if("WEIXIN".equals(type)){     // dosomething }else if("APP".equals(type)){     // dosomething }else if("WEB".equals(type)){     // dosomething }

現在使用枚舉來消除這些if/else。

寫一個加密用的接口,有加密和解密兩個方法。然后用不同的算法去實現這個接口完成加解密。

public interface Util {      // 解密     String decrypt();      // 加密     String encrypt(); }

創建一個枚舉類來實現這個接口

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后,直接調用解密方法就行了。

String decryptMessage = UtilEnum.valueOf(type).decrypt();

以后,如果新增了一個其他加密方式,只需要修改上面的枚舉類就完成了,業務代碼都不需要改動。

這就是枚舉類比較高級的兩個用法。

“Java枚舉類的使用方法是什么”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

宜川县| 镇远县| 翁源县| 马尔康县| 轮台县| 准格尔旗| 肃南| 广饶县| 定远县| 贡觉县| 上犹县| 睢宁县| 洪江市| 酉阳| 安塞县| 湖南省| 牙克石市| 西宁市| 新龙县| 宁武县| 卢湾区| 都昌县| 上蔡县| 卓资县| 收藏| 长丰县| 海安县| 绥德县| 博乐市| 临武县| 偃师市| 永济市| 靖州| 加查县| 公安县| 四子王旗| 化德县| 贡山| 岳西县| 徐水县| 济源市|