三分鐘快速掌握J(rèn)ava中枚舉(enum)
什么是枚舉?
枚舉是JDK5引入的新特性。在某些情況下,一個(gè)類的對(duì)象是固定的,就可以定義為枚舉。在實(shí)際使用中,枚舉類型也可以作為一種規(guī)范,保障程序參數(shù)安全。枚舉有以下特點(diǎn):
- Java中枚舉和類、接口的級(jí)別相同。
- 枚舉和類一樣,都有自己的屬性、方法、構(gòu)造方法,不同點(diǎn)是:枚舉的構(gòu)造方法只能是private修飾,也就無法從外部構(gòu)造對(duì)象。構(gòu)造方法只在構(gòu)造枚舉值時(shí)調(diào)用。
- 使用enum關(guān)鍵字聲明一個(gè)枚舉類型時(shí),就默認(rèn)繼承自Java中的
java.lang.Enum
類,并實(shí)現(xiàn)了java.lang.Seriablizable
和java.lang.Comparable
兩個(gè)接口。
- 所有的枚舉值都是
public static final
的,且非抽象的枚舉類不能再派生子類。
- 枚舉類的所有實(shí)例(枚舉值)必須在枚舉類的第一行顯式地列出,否則這個(gè)枚舉類將永遠(yuǎn)不能產(chǎn)生實(shí)例。
- 判斷枚舉是否相同時(shí),使用 == 和 equals 是一樣的。
下面是 java.lang.Enum
類中的 equals()
:
// 這里是final修飾的,不允許子類重寫 public final boolean equals(Object other) { return this==other; }
枚舉的常用方法
int compareTo(E o)
比較此枚舉與指定對(duì)象的順序。在該對(duì)象小于、等于或大于指定對(duì)象時(shí),分別返回負(fù)整數(shù)、零或正整數(shù)。 枚舉常量只能與相同枚舉類型的其他枚舉常量進(jìn)行比較。
// Enum 中的源碼 public final int compareTo(E o) { Enum other = (Enum)o; Enum self = this; if (self.getClass() != other.getClass() && // optimization self.getDeclaringClass() != other.getDeclaringClass()) throw new ClassCastException(); return self.ordinal - other.ordinal; }
String name()
返回此枚舉實(shí)例的名稱。
static values()
返回一個(gè)包含全部枚舉值的數(shù)組,可以用來遍歷所有枚舉值。
String toString()
返回此枚舉實(shí)例的名稱,即枚舉值。與 name()
一樣。
// Enum 中 name() 和 toString() public String toString() { return name; } public final String name() { return name; }
int ordinal()
返回枚舉值在枚舉類中的索引值(從0開始),即枚舉值在枚舉聲明中的順序,這個(gè)順序根據(jù)枚舉值聲明的順序而定。
<T extends Enum<T>> valueOf()
返回帶指定名稱的指定枚舉類型的枚舉常量,名稱必須與在此類型中聲明枚舉常量所用的標(biāo)識(shí)符完全匹配(不允許使用額外的空白字符)。這個(gè)方法與toString相對(duì)應(yīng),因此重寫 toString()
方法,一定要重寫 valueOf()
方法(我們可以重寫 toString()
方法,但不能自己重寫 valueOf()
方法,當(dāng)我們重寫 toString()
方法時(shí),valueOf()
方法會(huì)自動(dòng)重寫,不用我們理會(huì)。)
枚舉的應(yīng)用
枚舉是一種特殊的類型,其用法和普通的類使用非常相似。
代替一組常量
public enum Color { RED, GREEN, BLANK, YELLOW }
switch 語(yǔ)句中使用
// JDK1.6 中switch加入了對(duì)枚舉的支持 enum Signal { GREEN, YELLOW, RED } ... switch (color) { case RED: color = Signal.GREEN; break; } ...
向枚舉中添加方法
public enum Color { RED("紅色"), GREEN("綠色"), BLANK("白色"), YELLO("黃色"); // 成員變量 private String name; // 構(gòu)造方法 private Color(String name) { this.name = name; } // get set 方法 public String getName() { return name; } public void setName(String name) { this.name = name; } }
實(shí)現(xiàn)接口
public interface Behaviour { void print(); } public enum Color implements Behaviour{ RED("紅色", 1), GREEN("綠色", 2), BLANK("白色", 3), YELLO("黃色", 4); //接口方法 @Override public void print() { System.out.println(this.index+":"+this.name); } }
包含抽象方法的枚舉類
public enum Operation { // 用于執(zhí)行加法運(yùn)算 PLUS { // 花括號(hào)部分其實(shí)是一個(gè)匿名內(nèi)部子類 @Override public double calculate(double x, double y) { return x + y; } }, // 用于執(zhí)行減法運(yùn)算 MINUS { // 花括號(hào)部分其實(shí)是一個(gè)匿名內(nèi)部子類 @Override public double calculate(double x, double y) { // TODO Auto-generated method stub return x - y; } }, // 用于執(zhí)行乘法運(yùn)算 TIMES { // 花括號(hào)部分其實(shí)是一個(gè)匿名內(nèi)部子類 @Override public double calculate(double x, double y) { return x * y; } }, // 用于執(zhí)行除法運(yùn)算 DIVIDE { // 花括號(hào)部分其實(shí)是一個(gè)匿名內(nèi)部子類 @Override public double calculate(double x, double y) { return x / y; } }; //為該枚舉類定義一個(gè)抽象方法,枚舉類中所有的枚舉值都必須實(shí)現(xiàn)這個(gè)方法 public abstract double calculate(double x, double y); }
使用枚舉實(shí)現(xiàn)單例(單例的最佳實(shí)踐)
好處:
1.利用的枚舉的特性實(shí)現(xiàn)單例
2.由JVM保證線程安全
3.序列化和反射攻擊已經(jīng)被枚舉解決
public enum Singleton { INSTANCE; public Singleton getInstance(){ // 增加這個(gè)方法是讓別人明白怎么使用,因?yàn)檫@種實(shí)現(xiàn)方式還比較少見。 return INSTANCE; } }
其他關(guān)于枚舉的使用
EnumSet
range(E from, E to)
從枚舉值中獲取一段范圍的 Set。
for(WeekDayEnum day : EnumSet.range(WeekDayEnum.Mon, WeekDayEnum.Fri)) { System.out.println(day); }
of(E first, E... rest)
創(chuàng)建一個(gè)最初包含指定元素的枚舉 Set。
noneOf(Class<E> elementType)
創(chuàng)建一個(gè)具有指定元素類型的空枚舉 Set。
EnumMap
EnumMap(Class<K> keyType)
創(chuàng)建一個(gè)具有指定鍵類型的空枚舉Map。
Map<Weather, String> enumMap = new EnumMap<Weather, String>(Weather.class); enumMap.put(Weather.Sunny, "晴天"); enumMap.put(Weather.Rainy, "雨天");
Android中的枚舉
Enum 需要占用較大的內(nèi)存,如果對(duì)內(nèi)存敏感,請(qǐng)盡量少使用 Enum,換成靜態(tài)常量。
但是如果不使用枚舉,會(huì)出現(xiàn)一些安全隱患,所以官方推出了兩個(gè)注解,可以在編譯時(shí)期進(jìn)行類型檢查,以此替代枚舉。這兩個(gè)注解分別是:@IntDef 和 @StringDef。位于compile 'com.android.support:support-annotations:+' 。
使用示例
@StringDef的使用與 @IntDef一致,這里以 @IntDef為例。
public interface QRCodeType { int WECHAT = 0; int ALIPAY = 1; @IntDef({WECHAT , ALIPAY }) @Retention(RetentionPolicy.SOURCE) @Target({ElementType.FIELD, ElementType.METHOD, ElementType.PARAMETER}) @interface Checker { } } public class QRCode { @QRCodeType.Checker // 定義在屬性 private int type; public void setType(@QRCodeType.Checker int type) { // 定義在參數(shù) this.type= type; } @QRCodeType.Checker // 定義在方法(也就是檢查返回值的類型) public int getType() { return type; } }
使用建議
開發(fā)中使用范圍最廣的就是利用枚舉代替一組靜態(tài)常量,這種情況可以使用以上注解方式替代。
當(dāng)枚舉還含有其它功能時(shí)(如:包含其它定義的方法),則不能替換。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。
相關(guān)文章
關(guān)于SpringBoot整合Canal數(shù)據(jù)同步的問題
大家都知道canal是阿里巴巴旗下的一款開源工具,純java開發(fā),支持mysql數(shù)據(jù)庫(kù),本文給大家介紹SpringBoot整合Canal數(shù)據(jù)同步的問題,需要的朋友可以參考下2022-03-03idea創(chuàng)建maven項(xiàng)目速度慢的三種解決方案
這篇文章主要介紹了idea創(chuàng)建maven項(xiàng)目速度慢的三種解決方案,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2019-01-01springboot向elk寫日志實(shí)現(xiàn)過程
這篇文章主要介紹了springboot向elk寫日志實(shí)現(xiàn)過程,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10spring?boot?executable?jar/war?原理解析
spring boot里其實(shí)不僅可以直接以 java -jar demo.jar的方式啟動(dòng),還可以把jar/war變?yōu)橐粋€(gè)可以執(zhí)行的腳本來啟動(dòng),比如./demo.jar,這篇文章主要介紹了spring?boot?executable?jar/war?原理,需要的朋友可以參考下2023-02-02詳解java如何處理各種批量數(shù)據(jù)入庫(kù)
這篇文章主要為大家詳細(xì)介紹了java如何使用BlockingQueue處理各種批量數(shù)據(jù)入庫(kù),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-11-11RabbitMQ延遲隊(duì)列及消息延遲推送實(shí)現(xiàn)詳解
這篇文章主要介紹了RabbitMQ延遲隊(duì)列及消息延遲推送實(shí)現(xiàn)詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12