亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

關于Java中枚舉Enum的深入剖析

2019-11-14 10:06:52
字體:
來源:轉載
供稿:網友

學習java的同學注意了!!! 學習過程中遇到什么問題或者想獲取學習資源的話,歡迎加入Java學習交流群,群號碼:183993990  我們一起學Java!

在編程語言中我們,都會接觸到枚舉類型,通常我們進行有窮的列舉來實現一些限定。Java也不例外。Java中的枚舉類型為Enum,本文將對枚舉進行一些比較深入的剖析。

什么是Enum

Enum是自Java 5 引入的特性,用來方便Java開發者實現枚舉應用。一個簡單的Enum使用如下。

// ColorEnum.javapublic enum ColorEmun {    RED,    GREEN,    YELLOW}public void setColorEnum(ColorEmun colorEnum) {    //some code here}setColorEnum(ColorEmun.GREEN);

為什么會有Enum

在Enum之前的我們使用類似如下的代碼實現枚舉的功能.

public static final int COLOR_RED = 0;public static final int COLOR_GREEN = 1;public static final int COLOR_YELLOW = 2;public void setColor(int color) {    //some code here}//調用setColor(COLOR_RED)

然而上面的還是有不盡完美的地方

setColor(COLOR_RED)與setColor(0)效果一樣,而后者可讀性很差,但卻可以正常運行setColor方法可以接受枚舉之外的值,比如setColor(3),這種情況下程序可能出問題

概括而言,傳統枚舉有如下兩個弊端

安全性可讀性,尤其是打印日志時

因此Java引入了Enum,使用Enum,我們實現上面的枚舉就很簡單了,而且還可以輕松避免傳入非法值的風險.

枚舉原理是什么

Java中Enum的本質其實是在編譯時期轉換成對應的類的形式。

首先,為了探究枚舉的原理,我們先簡單定義一個枚舉類,這里以季節為例,類名為Season,包含春夏秋冬四個枚舉條目.

public enum Season {    SPRING,    SUMMER,    AUTUMN,    WINTER}

然后我們使用javac編譯上面的類,得到class文件.

javac Season.java

然后,我們利用反編譯的方法來看看字節碼文件究竟是什么.這里使用的工具是javap的簡單命令,先列舉一下這個Season下的全部元素.

company javap SeasonWarning: Binary file Season contains com.company.SeasonCompiled from "Season.java"public final class com.company.Season extends java.lang.Enum<com.company.Season> {  public static final com.company.Season SPRING;  public static final com.company.Season SUMMER;  public static final com.company.Season AUTUMN;  public static final com.company.Season WINTER;  public static com.company.Season[] values();  public static com.company.Season valueOf(java.lang.String);  static {};}

從上反編譯結果可知

java代碼中的Season轉換成了繼承自的java.lang.enum的類既然隱式繼承自java.lang.enum,也就意味java代碼中,Season不能再繼承其他的類Season被標記成了final,意味著它不能被繼承

static代碼塊

使用javap具體反編譯class文件,得到靜態代碼塊相關的結果為

static {};    Code:       0: new           #4                  // class com/company/Season       3: dup       4: ldc           #7                  // String SPRING       6: iconst_0       7: invokespecial #8                  // Method "<init>":(Ljava/lang/String;I)V      10: putstatic     #9                  // Field SPRING:Lcom/company/Season;      13: new           #4                  // class com/company/Season      16: dup      17: ldc           #10                 // String SUMMER      19: iconst_1      20: invokespecial #8                  // Method "<init>":(Ljava/lang/String;I)V      23: putstatic     #11                 // Field SUMMER:Lcom/company/Season;      26: new           #4                  // class com/company/Season      29: dup      30: ldc           #12                 // String AUTUMN      32: iconst_2      33: invokespecial #8                  // Method "<init>":(Ljava/lang/String;I)V      36: putstatic     #13                 // Field AUTUMN:Lcom/company/Season;      39: new           #4                  // class com/company/Season      42: dup      43: ldc           #14                 // String WINTER      45: iconst_3      46: invokespecial #8                  // Method "<init>":(Ljava/lang/String;I)V      49: putstatic     #15                 // Field WINTER:Lcom/company/Season;      52: iconst_4      53: anewarray     #4                  // class com/company/Season      56: dup      57: iconst_0      58: getstatic     #9                  // Field SPRING:Lcom/company/Season;      61: aastore      62: dup      63: iconst_1      64: getstatic     #11                 // Field SUMMER:Lcom/company/Season;      67: aastore      68: dup      69: iconst_2      70: getstatic     #13                 // Field AUTUMN:Lcom/company/Season;      73: aastore      74: dup      75: iconst_3      76: getstatic     #15                 // Field WINTER:Lcom/company/Season;      79: aastore      80: putstatic     #1                  // Field $VALUES:[Lcom/company/Season;      83: return}

其中

0~52為實例化SPRING, SUMMER, AUTUMN, WINTER53~83為創建Season[]數組$VALUES,并將上面的四個對象放入數組的操作.

values方法

values方法的的返回值實際上就是上面$VALUES數組對象

swtich中的枚舉

在Java中,switch-case是我們經常使用的流程控制語句.當枚舉出來之后,switch-case也很好的進行了支持.

比如下面的代碼是完全正常編譯,正常運行的.

public static void main(String[] args) {        Season season = Season.SPRING;        switch(season) {            case SPRING:                System.out.println("It's Spring");                break;            case WINTER:                System.out.println("It's Winter");                break;            case SUMMER:                System.out.println("It's Summer");                break;            case AUTUMN:                System.out.println("It's Autumn");                break;        }    }

不過,通常情況下switch-case支持類似int的類型,那么它是怎么做到對Enum的支持呢,我們反編譯上述方法看一下字節碼的真實情況.

public static void main(java.lang.String[]);    Code:       0: getstatic     #2                  // Field com/company/Season.SPRING:Lcom/company/Season;       3: astore_1       4: getstatic     #3                  // Field com/company/Main$1.$SwitchMap$com$company$Season:[I       7: aload_1       8: invokevirtual #4                  // Method com/company/Season.ordinal:()I      11: iaload      12: tableswitch   { // 1 to 4                     1: 44                     2: 55                     3: 66                     4: 77               default: 85          }      44: getstatic     #5                  // Field java/lang/System.out:Ljava/io/PrintStream;      47: ldc           #6                  // String It's Spring      49: invokevirtual #7                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V      52: goto          85      55: getstatic     #5                  // Field java/lang/System.out:Ljava/io/PrintStream;      58: ldc           #8                  // String It's Winter      60: invokevirtual #7                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V      63: goto          85      66: getstatic     #5                  // Field java/lang/System.out:Ljava/io/PrintStream;      69: ldc           #9                  // String It's Summer      71: invokevirtual #7                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V      74: goto          85      77: getstatic     #5                  // Field java/lang/System.out:Ljava/io/PrintStream;      80: ldc           #10                 // String It's Autumn      82: invokevirtual #7                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V      88: return

注意上面代碼塊有這樣的一段代碼

8: invokevirtual #4                  // Method com/company/Season.ordinal:()I

事實果真如此,在switch-case中,還是將Enum轉成了int值(通過調用Enum.oridinal()方法)

枚舉與混淆

在Android開發中,進行混淆是我們在發布前必不可少的工作,混下后,我們能增強反編譯的難度,在一定程度上保護了增強了安全性.

而開發人員處理混淆更多的是將某些元素加入不混淆的名單,這里枚舉就是需要排除混淆的.

在默認的混淆配置文件中,已經加入了關于對枚舉混淆的處理

# For enumeration classes, see http://proguard.sourceforge.net/manual/examples.html#enumerations-keepclassmembers enum * {    public static **[] values();    public static ** valueOf(java.lang.String);}

關于為什么要保留values()方法和valueOf()方法,請參考文章讀懂 Android 中的代碼混淆 關于枚舉的部分

使用proguard優化

使用Proguard進行優化,可以將枚舉盡可能的轉換成int。配置如下

-optimizations class/unboxing/enum

確保上述代碼生效,需要確proguard配置文件不包含-dontoptimize指令。

當我們使用gradlew打包是,看到類似下面的輸出,即Number of unboxed enum classes:1代表已經將一個枚舉轉換成了int的形式。

Optimizing...  Number of finalized classes:                 0   (disabled)  Number of unboxed enum classes:              1  Number of vertically merged classes:         0   (disabled)  Number of horizontally merged classes:       0   (disabled)

枚舉單例

單例模式是我們在日常開發中可謂是最常用的設計模式.

然后要設計好單例模式,無非考慮一下幾點

確保只有唯一實例,不多創建多余實例確保實例按需創建.

因此傳統的做法想要實現單例,大致有一下幾種

餓漢式加載懶漢式synchronize和雙重檢查利用java的靜態加載機制

相比上述的方法,使用枚舉也可以實現單例,而且還更加簡單.

public enum AppManager {    INSTANCE;    private String tagName;    public void setTag(String tagName) {        this.tagName = tagName;    }    public String getTag() {        return tagName;    }}

調用起來也更加簡單

AppManager.INSTANCE.getTag();

枚舉如何確保唯一實例

因為獲得實例只能通過AppManager.INSTANCE

下面的方式是不可以的

AppManager appManager = new AppManager(); //compile error

關于單例模式,可以閱讀單例這種設計模式了解更多。

(Android中)該不該用枚舉

既然上面提到了枚舉會轉換成類,這樣理論上造成了下面的問題

增加了dex包的大小,理論上dex包越大,加載速度越慢同時使用枚舉,運行時的內存占用也會相對變大

關于上面兩點的驗證,秋百萬已經做了詳細的論證,大家可以參考這篇文章《Android 中的 Enum 到底占多少內存?該如何用?》

關于枚舉是否使用的結論,大家可以參考

如果你開發的是Framework不建議使用enum如果是簡單的enum,可以使用int很輕松代替,則不建議使用enum另外,如果是Android中,可以使用下面介紹的枚舉注解來實現。除此之外,我們還需要對比可讀性和易維護性來與性能進行衡量,從中進行做出折中

在Android中的替代

Android中新引入的替代枚舉的注解有IntDef和StringDef,這里以IntDef做例子說明一下.

public class Colors {    @IntDef({RED, GREEN, YELLOW})    @Retention(RetentionPolicy.SOURCE)    public @interface LightColors{}    public static final int RED = 0;    public static final int GREEN = 1;    public static final int YELLOW = 2;}聲明必要的int常量聲明一個注解為LightColors使用@IntDef修飾LightColors,參數設置為待枚舉的集合使用@Retention(RetentionPolicy.SOURCE)指定注解僅存在與源碼中,不加入到class文件中

比如我們用來標注方法的參數

private void setColor(@Colors.LightColors int color) {        Log.d("MainActivity", "setColor color=" + color);}

調用的該方法的時候

setColor(Colors.GREEN);

以上就是我對Java中enum的一些深入的剖析,歡迎大家不吝賜教。

學習Java的同學注意了!?。?nbsp;學習過程中遇到什么問題或者想獲取學習資源的話,歡迎加入Java學習交流群,群號碼:183993990  我們一起學Java!


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久福利视频导航| 久久久久久久香蕉网| 国产精品国产三级国产aⅴ浪潮| 亚洲精品720p| 亚洲欧美自拍一区| 亚洲精品国偷自产在线99热| 日韩欧美国产免费播放| 久久香蕉国产线看观看av| 国产精品扒开腿做| 亚洲的天堂在线中文字幕| 国产精品爱啪在线线免费观看| 日日噜噜噜夜夜爽亚洲精品| 日韩中文字幕在线看| 在线亚洲国产精品网| 日韩精品极品视频免费观看| 91在线视频导航| 亚洲国产成人av在线| 国产成人一区二| 国产精品久久久久久婷婷天堂| 亚洲男人的天堂在线| 国产一区二区丝袜| 91在线视频免费| 国产一区二区三区在线视频| 高跟丝袜一区二区三区| 久久久久久九九九| 国产欧美精品一区二区三区介绍| 正在播放欧美视频| 日韩av在线免播放器| 久久天天躁狠狠躁夜夜爽蜜月| 久久久久久噜噜噜久久久精品| 久久香蕉国产线看观看av| 国产精品男人爽免费视频1| 青青青国产精品一区二区| 久久久久999| 91香蕉嫩草神马影院在线观看| 亚洲精品电影网在线观看| 6080yy精品一区二区三区| 日韩有码片在线观看| 在线看日韩av| 久久成人一区二区| 亚洲国产97在线精品一区| 亚洲自拍偷拍网址| 国产97在线亚洲| 91精品视频在线| 亚洲社区在线观看| 亚洲国产日韩一区| 日本一区二三区好的精华液| 国产一区二区三区中文| 欧美电影免费在线观看| 九九久久国产精品| 精品亚洲一区二区| 深夜福利亚洲导航| 亚洲精品短视频| 免费91麻豆精品国产自产在线观看| 91av在线视频观看| 日韩av手机在线| 欧美一级免费看| 国产不卡视频在线| 国产精品久久9| 久久精品国产99国产精品澳门| 欧美又大粗又爽又黄大片视频| 亚洲国产精品va在线观看黑人| 亚洲肉体裸体xxxx137| 欧美日韩国产综合视频在线观看中文| 国产精品九九久久久久久久| 在线看日韩av| 国产精品高潮视频| 亚洲黄页网在线观看| 亚洲自拍另类欧美丝袜| 国产suv精品一区二区三区88区| 狠狠久久五月精品中文字幕| 亚洲综合日韩中文字幕v在线| 国产精品白嫩初高中害羞小美女| 精品露脸国产偷人在视频| 国产精品欧美激情在线播放| 欧美日韩中文在线观看| 国产伊人精品在线| 日韩中文字幕免费视频| 日韩国产高清视频在线| 亚洲国产欧美一区| 亚洲精品中文字幕有码专区| 88国产精品欧美一区二区三区| 久久99国产精品自在自在app| 欧美日韩成人黄色| 久久偷看各类女兵18女厕嘘嘘| 色综合色综合久久综合频道88| 久久久久九九九九| 亚洲国产精品电影| 日韩av在线导航| 欧美专区在线播放| 亚洲第一福利在线观看| 91精品国产91久久久久久久久| 国产三级精品网站| 欧美xxxx18国产| 欧美寡妇偷汉性猛交| 日韩在线视频线视频免费网站| 色偷偷av一区二区三区乱| 日韩一级黄色av| 久久久精品美女| 国产精品嫩草影院一区二区| 久久影院中文字幕| 国产精品毛片a∨一区二区三区|国| 亚洲春色另类小说| 一区二区三区www| 欧美日韩国产成人在线| 九色91av视频| 欧美高清理论片| 超碰91人人草人人干| 久久久综合av| 欧美丰满少妇xxxxx做受| 中文字幕欧美日韩va免费视频| 成人午夜在线影院| 欧美日本亚洲视频| 人九九综合九九宗合| 久久久久国产精品免费网站| 欧美日韩国产中文精品字幕自在自线| 日韩精品中文字幕久久臀| 国产日本欧美一区二区三区在线| 亚洲男人av在线| 欧美视频第一页| 91系列在线播放| 国内精品小视频在线观看| 欧美极品少妇xxxxⅹ免费视频| 国产成人精品免高潮在线观看| 欧美成人自拍视频| 狠狠躁夜夜躁人人爽天天天天97| 欧美成人免费在线观看| 欧美国产日韩免费| 国产精品视频久久久| 午夜精品久久久久久久久久久久| 这里只有精品丝袜| 88国产精品欧美一区二区三区| 北条麻妃一区二区三区中文字幕| 最新国产成人av网站网址麻豆| 国产精品久久综合av爱欲tv| 亚洲精品视频免费在线观看| 国产精品自在线| 自拍偷拍亚洲精品| 欧美性xxxxxx| 亚洲第一av网| 精品久久久久久亚洲国产300| 日韩美女免费线视频| www.日本久久久久com.| 亚洲成人精品视频| 国产日本欧美一区二区三区| 成人h视频在线| 最近2019中文字幕在线高清| 国产精品丝袜一区二区三区| 综合国产在线观看| 国产精品成人久久久久| 福利视频第一区| 2021久久精品国产99国产精品| 亚洲精品网址在线观看| 国产精品直播网红| 欧美国产日本在线| 欧美亚洲第一页| 2021久久精品国产99国产精品| www.欧美精品| 尤物yw午夜国产精品视频明星| 精品日本美女福利在线观看| 性欧美在线看片a免费观看| 亚洲天堂成人在线视频| 国模极品一区二区三区| 亚洲爱爱爱爱爱|