java-Enum

轉-https://www.cnblogs.com/liaojie970/p/6474733.htmlhtml

Java Enum原理 java

public enum Size{ SMALL, MEDIUM, LARGE, EXTRA_LARGE };

實際上,這個聲明定義的類型是一個類,它恰好有四個實例,在此儘可能不要構造新對象。程序員

所以,在比較兩個枚舉類型的值時,永遠不須要調用equals方法,而直接使用"=="就能夠了。(equals()方法也是直接使用==,  二者是同樣的效果)設計模式

Java Enum類型的語法結構儘管和java類的語法不同,應該說差異比較大。可是通過編譯器編譯以後產生的是一個class文件。該class文件通過反編譯能夠看到其實是生成了一個類,該類繼承了java.lang.Enum<E>。數組

  例如:ide

複製代碼
public enum WeekDay { Mon("Monday"), Tue("Tuesday"), Wed("Wednesday"), Thu("Thursday"), Fri( "Friday"), Sat("Saturday"), Sun("Sunday"); private final String day; private WeekDay(String day) { this.day = day; } public static void printDay(int i){ switch(i){ case 1: System.out.println(WeekDay.Mon); break; case 2: System.out.println(WeekDay.Tue);break; case 3: System.out.println(WeekDay.Wed);break; case 4: System.out.println(WeekDay.Thu);break; case 5: System.out.println(WeekDay.Fri);break; case 6: System.out.println(WeekDay.Sat);break; case 7: System.out.println(WeekDay.Sun);break; default:System.out.println("wrong number!"); } } public String getDay() { return day; } }
複製代碼

WeekDay通過反編譯(javap WeekDay命令)以後獲得的內容以下(去掉了彙編代碼):函數

複製代碼
public final class WeekDay extends java.lang.Enum{ public static final WeekDay Mon; public static final WeekDay Tue; public static final WeekDay Wed; public static final WeekDay Thu; public static final WeekDay Fri; public static final WeekDay Sat; public static final WeekDay Sun; static {}; public static void printDay(int); public java.lang.String getDay(); public static WeekDay[] values(); public static WeekDay valueOf(java.lang.String); }
複製代碼

用法一:常量post

在JDK1.5 以前,咱們定義常量都是: public static fianl.... 。如今好了,有了枚舉,能夠把相關的常量分組到一個枚舉類型裏,並且枚舉提供了比常量更多的方法。this

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

用法二:switchspa

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 class Test { 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; } } public static void main(String[] args) { System.out.println(Color.RED.toString()); } }
複製代碼

用法五:實現接口

全部的枚舉都繼承自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 } }
複製代碼

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

java.util.EnumSet和java.util.EnumMap是兩個枚舉集合。EnumSet保證集合中的元素不重複;EnumMap中的 key是enum類型,而value則能夠是任意類型。關於這個兩個集合的使用就不在這裏贅述,

能夠參考JDK文檔

3、 完整示例代碼

枚舉類型的完整演示代碼以下:

複製代碼
public class LightTest { // 1.定義枚舉類型 public enum Light { // 利用構造函數傳參  RED(1), GREEN(3), YELLOW(2); // 定義私有變量 private int nCode; // 構造函數,枚舉類型只能爲私有 private Light(int _nCode) { this.nCode = _nCode; } @Override public String toString() { return String.valueOf(this.nCode); } } /** * * @param args */ public static void main(String[] args) { // 1.遍歷枚舉類型  System.out.println("演示枚舉類型的遍歷 ......"); testTraversalEnum(); // 2.演示EnumMap對象的使用  System.out.println("演示EnmuMap對象的使用和遍歷....."); testEnumMap(); // 3.演示EnmuSet的使用  System.out.println("演示EnmuSet對象的使用和遍歷....."); testEnumSet(); } /** * * 演示枚舉類型的遍歷 */ private static void testTraversalEnum() { Light[] allLight = Light.values(); for (Light aLight : allLight) { System.out.println("當前燈name:" + aLight.name()); System.out.println("當前燈ordinal:" + aLight.ordinal()); System.out.println("當前燈:" + aLight); } } /** * * 演示EnumMap的使用,EnumMap跟HashMap的使用差很少,只不過key要是枚舉類型 */ private static void testEnumMap() { // 1.演示定義EnumMap對象,EnumMap對象的構造函數須要參數傳入,默認是key的類的類型  EnumMap<Light, String> currEnumMap = new EnumMap<Light, String>( Light.class); currEnumMap.put(Light.RED, "紅燈"); currEnumMap.put(Light.GREEN, "綠燈"); currEnumMap.put(Light.YELLOW, "黃燈"); // 2.遍歷對象 for (Light aLight : Light.values()) { System.out.println("[key=" + aLight.name() + ",value=" + currEnumMap.get(aLight) + "]"); } } /** * * 演示EnumSet如何使用,EnumSet是一個抽象類,獲取一個類型的枚舉類型內容<BR/> * * 可使用allOf方法 */ private static void testEnumSet() { EnumSet<Light> currEnumSet = EnumSet.allOf(Light.class); for (Light aLightSetElement : currEnumSet) { System.out.println("當前EnumSet中數據爲:" + aLightSetElement); } } }
複製代碼

執行結果以下:

複製代碼
演示枚舉類型的遍歷 ......

當前燈name:RED

當前燈ordinal:0 當前燈:1 當前燈name:GREEN 當前燈ordinal:1 當前燈:3 當前燈name:YELLOW 當前燈ordinal:2 當前燈:2 演示EnmuMap對象的使用和遍歷..... [key=RED,value=紅燈] [key=GREEN,value=綠燈] [key=YELLOW,value=黃燈] 演示EnmuSet對象的使用和遍歷..... 當前EnumSet中數據爲:1 當前EnumSet中數據爲:3 當前EnumSet中數據爲:2
複製代碼

 

1. 全部的枚舉類型都是Enum類的子類。 它們繼承了這個類的許多方法。其中最有用的一個方法是toString(),這個方法可以返回枚舉常量名。   toString()方法的逆方法是靜態方法valueOf(Class, String). 例如 Light lt = (Light) Enum.valueOf(Light.class, "RED"); 將lt設置爲 Light.RED。 每一個枚舉類型都有一個靜態的values()方法,它將返回一個包含所有枚舉值的數組。   ordinal()方法返回enum聲明中枚舉常量的位置,位置從0開始計數。例如  Light.GREEN.ordinal()返回1。   Enum類實現了Comparable接口,  int  compareTo( E other)  若是枚舉常量在other以前,則返回一個負值; 若是this==other,則返回0;不然,返回正值。 枚舉常量的出現次序在enum 聲明中給出。(因此不能直接用<,>符號比較兩個枚舉值)

2. 能夠建立一個enum類,把它看作一個普通的類。除了它不能繼承其餘類了。(java是單繼承,它已經繼承了Enum),

能夠添加其餘方法,覆蓋它自己的方法

3. switch()參數可使用enum了

4. values()方法是編譯器插入到enum定義中的static方法,因此,當你將enum實例向上轉型爲父類Enum是,values()就不可訪問了。解決辦法:在Class中有一個getEnumConstants()方法,因此即使Enum接口中沒有values()方法,咱們仍然能夠經過Class對象取得全部的enum實例

5. 沒法從enum繼承子類,若是須要擴展enum中的元素,在一個接口的內部,建立實現該接口的枚舉,以此將元素進行分組。達到將枚舉元素進行分組。

6. 使用EnumSet代替標誌。enum要求其成員都是惟一的,可是enum中不能刪除添加元素。

7. EnumMap的key是enum,value是任何其餘Object對象。

8. enum容許程序員爲eunm實例編寫方法。因此能夠爲每一個enum實例賦予各自不一樣的行爲。

9. 使用enum的職責鏈(Chain of Responsibility) .這個關係到設計模式的職責鏈模式。以多種不一樣的方法來解決一個問題。而後將他們連接在一塊兒。當一個請求到來時,遍歷這個鏈,直到鏈中的某個解決方案可以處理該請求。

10. 使用enum的狀態機

11. 使用enum多路分發

相關文章
相關標籤/搜索