K:java中枚舉的常見用法

来源:http://www.cnblogs.com/MyStringIsNotNull/archive/2017/12/10/8018189.html
-Advertisement-
Play Games

用法一:常量   在JDK1.5 之前,我們定義常量都是: public static fianl....。現在好了,有了枚舉,可以把相關的常量分組到一個枚舉類型里,而且枚舉提供了比常量更多的方法。 用法二:switch   JDK1.6之前的switch語句 ...


用法一:常量

  在JDK1.5 之前,我們定義常量都是: public static fianl....。現在好了,有了枚舉,可以把相關的常量分組到一個枚舉類型里,而且枚舉提供了比常量更多的方法。

public enum Color {  
      RED, GREEN, BLANK, YELLOW  
    }  

用法二:switch

  JDK1.6之前的switch語句只支持int,char,enum類型,使用枚舉,能讓我們的代碼可讀性更強。

enum Signal {  
    GREEN, YELLOW, RED  
}  
public class TrafficLight {  
    Signal color = Signal.RED;  
    public void change() {  
        switch (color) {  
        case RED:  
            color = Signal.GREEN;  
            break;  
        case YELLOW:  
            color = Signal.RED;  
            break;  
        case GREEN:  
            color = Signal.YELLOW;  
            break;  
        }  
    }  
}  

用法三:向枚舉中添加新方法

  如果打算自定義自己的方法,那麼必須在enum實例序列的最後添加一個分號。而且Java要求必須先定義enum實例。

public enum Color {  
        RED("紅色", 1), GREEN("綠色", 2), BLANK("白色", 3), YELLO("黃色", 4);  
        // 成員變數  
        private String name;  
        private int index;  
        // 構造方法  
        private Color(String name, int index) {  
            this.name = name;  
            this.index = index;  
        }  
        // 普通方法  
        public static String getName(int index) {  
            for (Color c : Color.values()) {  
                if (c.getIndex() == index) {  
                    return c.name;  
                }  
            }  
            return null;  
        }  
        // get set 方法  
        public String getName() {  
            return name;  
        }  
        public void setName(String name) {  
            this.name = name;  
        }  
        public int getIndex() {  
            return index;  
        }  
        public void setIndex(int index) {  
            this.index = index;  
        }  
    }  

用法四:覆蓋枚舉的方法

下麵給出一個toString()方法覆蓋的例子。

public enum Color {  
        RED("紅色", 1), GREEN("綠色", 2), BLANK("白色", 3), YELLO("黃色", 4);  
        // 成員變數  
        private String name;  
        private int index;  
        // 構造方法  
        private Color(String name, int index) {  
            this.name = name;  
            this.index = index;  
        }  
        //覆蓋方法  
        @Override  
        public String toString() {  
            return this.index+"_"+this.name;  
        }  
    }  

用法五:實現介面

  所有的枚舉都繼承自java.lang.Enum類。由於Java不支持多繼承,所以枚舉對象不能再繼承其他類。

public interface Behaviour {  
        void print();  
        String getInfo();  
    }  
    public enum Color implements Behaviour{  
        RED("紅色", 1), GREEN("綠色", 2), BLANK("白色", 3), YELLO("黃色", 4);  
        // 成員變數  
        private String name;  
        private int index;  
        // 構造方法  
        private Color(String name, int index) {  
            this.name = name;  
            this.index = index;  
        }  
    //介面方法  
        @Override  
        public String getInfo() {  
            return this.name;  
        }  
        //介面方法  
        @Override  
        public void print() {  
            System.out.println(this.index+":"+this.name);  
        }  
    }  

用法六:使用介面組織枚舉

public interface Food {  
        enum Coffee implements Food{  
            BLACK_COFFEE,DECAF_COFFEE,LATTE,CAPPUCCINO  
        }  
        enum Dessert implements Food{  
            FRUIT, CAKE, GELATO  
        }  
    }  

用法七:關於枚舉集合的使用

如下內容有用到的代碼:

public enum EnumTest01 {  
      
    UPDATE(1,"更新"),QUERY(2,"查詢"),DELETE(3,"刪除");  
    private Integer enumValue;  
    private String enumDesc;  
      
    private EnumTest01(Integer enumValue, String enumDesc) {  
        this.enumValue = enumValue;  
        this.enumDesc = enumDesc;  
    }  
      
    public int getEnumValue(){  
        return this.enumValue;  
    }  
      
    public String getEnumDesc(){  
        return this.enumDesc;  
    }  
}  

EnumMap

  Map的實現類有很多種,EnumMap從名字我們可以看出這個Map是給枚舉類用的。它的key為枚舉元素,value自定義。在工作中我們也可以用其他的Map來實現我們關於枚舉的需求,但是為什麼要用這個EnumMap呢?因為它的性能高!為什麼性能高?因為它的內部是用數組的數據結構來維護的!我們可以看一下它的源碼實現:

put方法

public V put(K key, V value) {  
        typeCheck(key);  
      
        int index = key.ordinal();  
        Object oldValue = vals[index];  
        vals[index] = maskNull(value);  
        if (oldValue == null)  
            size++;  
        return unmaskNull(oldValue);  
    }  

  typeCheck是用來檢查key的類型的,因為key只能為枚舉元素。接下來的這一句int index = key.ordinal(); key.ordinal()這個就是我們上面說的枚舉類型的序號,然後被當做數組的下標,放到vals這個數組裡。那麼get方法呢?

get方法

public V get(Object key) {  
        return (isValidKey(key) ?  
                unmaskNull(vals[((Enum<?>)key).ordinal()]) : null);  
    }  

註意這一句話:vals[((Enum<?>)key).ordinal()]。這個不就是取得下標,根據下標獲取數組中的值嗎?!

remove方法

public V remove(Object key) {  
        if (!isValidKey(key))  
            return null;  
        int index = ((Enum<?>)key).ordinal();  
        Object oldValue = vals[index];  
        vals[index] = null;  
        if (oldValue != null)  
            size--;  
        return unmaskNull(oldValue);  
    }  

remove方法的實現也是挺簡單的,就是把相應下標的元素變為null,等著GC回收。

EnumSet

  EnumSet這是一個用來操作Enum的集合,是一個抽象類,它有兩個繼承類:JumboEnumSet和RegularEnumSet。在使用的時候,需要制定枚舉類型。它的特點也是速度非常快,為什麼速度很快呢?因為每次add的時候,每個枚舉值只占一個長整型的一位。我們可以翻看源碼來看看它的實現:

add方法

public boolean add(E e) {  
    typeCheck(e);  
  
    long oldElements = elements;  
    elements |= (1L << ((Enum<?>)e).ordinal());  
    return elements != oldElements;  
}  

從中我們可以看出是先對一個長整型左移枚舉類型的序列數,然後再和長整型 或 。
ps:其中值得我們引起註意的代碼的實現是1L<<((Enum<?>)e).ordinal());這段代碼,其實現了每個元素僅占一位的效果。

of方法

of方法有好幾個重載的方法,它的作用是創建一個最初包含指定元素的枚舉 Set。

EnumSet<EnumTest01> enumSets = EnumSet.of(EnumTest01.DELETE);  

allOf

創建一個包含指定元素類型的所有元素的枚舉 Set。

EnumSet<EnumTest01> enumSets = EnumSet.allOf(EnumTest01.class);  

range方法

創建一個指定範圍的Set。

EnumSet<EnumTest01> enumSets = EnumSet.range(EnumTest01.DELETE,EnumTest01.UPDATE);  

noneOf方法

創建一個指定枚舉類型的空set。

EnumSet<EnumTest01> enumSet = EnumSet.noneOf(EnumTest01.class);  
enumSet.add(EnumTest01.DELETE);  
enumSet.add(EnumTest01.UPDATE);  
for (Iterator<EnumTest01> it = enumSet.iterator(); it.hasNext();) {  
    System.out.println(it.next().getEnumDesc());  
}  
for (EnumTest01 enumTest : enumSet) {  
    System.out.println(enumTest.getEnumDesc() + "  ..... ");  
}  

copyOf方法

創建一個set的並copy所傳入的集合中的枚舉元素。

EnumSet<EnumTest01> enumSets = EnumSet.copyOf(enumSet);  

回到目錄|·(工)·)


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、作用域public,private,protected,以及不寫時的區別 答: 區別如下: 2、Anonymous Inner Class (匿名內部類) 是否可以extends(繼承)其它類,是否可以implements(實現)interface(介面) 答: 匿名的內部類是沒有名字的內部類。 ...
  • 1. 列表list 1.1 切片# 定義一個list。list = [1, 2, 3, 4, 5] 從左往右讀取字元(預設步長為 1 )。如:list[-2:-1] # 返回一個list數據類型,[6]list[2] # 返回一個int數據類型, 3 從右往左讀取字元串(預設步長為 1 )。如:li ...
  • 在開發項目時,有時候需要用到上傳功能,比如頭像上傳等,其文件會保存到伺服器中。但是我發現在用eclipse做項目的過程中,每次重新部署項目,原來上傳的文件就會丟失。 其原因是因為每次項目修改後,eclipse會把我們放在工作空間workspace中的這個項目拷貝到伺服器下(如tomcat的webap ...
  • ...
  • 在學習有關java枚舉的時候,我們知道了所有的枚舉類型均是繼承自java.lang.Enum類的,且所有的枚舉常量均是該枚舉類型的一個對象,且對象名即為該枚舉常量的名稱。例子如下:源碼: 反編譯後的代碼: 在寫代碼的時候,由於輸入法切換的問題,發現枚舉常量的常量名稱居然是可以使用中文的。代碼如下: ...
  • 一、PTA實驗作業 題目1:查驗身份證 1. 本題PTA提交列表 2. 設計思路 3.代碼截圖 4.本題調試過程碰到問題及PTA提交列表情況說明。 部分正確 :將x改為大寫x 題目2:藏頭詩 1. 本題PTA提交列表 2. 設計思路 3.代碼截圖 4.本題調試過程碰到問題及PTA提交列表情況說明。 ...
  • 使用會話維持狀態 一、會話 為了實現關聯同一個用戶端的多個請求和這些請求之間數據的共用,需要用到會話,會話用於維持請求和請求之間的狀態。從伺服器的角度,當用戶的Web瀏覽器打開第一個鏈接到伺服器的套接字時請求就開始了,直到伺服器返回最後一個數據包並關閉鏈接是,該請求將結束。此時用戶瀏覽器和伺服器之間 ...
  •   枚舉是如何保證線程安全的且其在序列化和反序列化的操作中是單例的?   要想看源碼,首先得有一個類吧,那麼枚舉類型到底是什麼類呢?是enum嗎?答案很明顯不是,enum就和class一樣,只是一個關鍵字,他並不是一個類,那麼枚舉是由什麼類維護的呢,我們簡單的 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...