恕我直言怀偷,我懷疑你并不會(huì)用 Java 枚舉

開門見山地說吧,enum(枚舉)是 Java 1.5 時(shí)引入的關(guān)鍵字播玖,它表示一種特殊類型的類椎工,默認(rèn)繼承自 java.lang.Enum。

為了證明這一點(diǎn)蜀踏,我們來新建一個(gè)枚舉 PlayerType:

public enum PlayerType {
    TENNIS,
    FOOTBALL,
    BASKETBALL
}

兩個(gè)關(guān)鍵字帶一個(gè)類名维蒙,還有大括號(hào),以及三個(gè)大寫的單詞脓斩,但沒看到繼承 Enum 類澳疚鳌畴栖?別著急随静,心急吃不了熱豆腐啊。使用 JAD 查看一下反編譯后的字節(jié)碼,就一清二楚了燎猛。

public final class PlayerType extends Enum
{

    public static PlayerType[] values()
    {
        return (PlayerType[])$VALUES.clone();
    }

    public static PlayerType valueOf(String name)
    {
        return (PlayerType)Enum.valueOf(com/cmower/baeldung/enum1/PlayerType, name);
    }

    private PlayerType(String s, int i)
    {
        super(s, i);
    }

    public static final PlayerType TENNIS;
    public static final PlayerType FOOTBALL;
    public static final PlayerType BASKETBALL;
    private static final PlayerType $VALUES[];

    static 
    {
        TENNIS = new PlayerType("TENNIS", 0);
        FOOTBALL = new PlayerType("FOOTBALL", 1);
        BASKETBALL = new PlayerType("BASKETBALL", 2);
        $VALUES = (new PlayerType[] {
            TENNIS, FOOTBALL, BASKETBALL
        });
    }
}

看到?jīng)]恋捆?PlayerType 類是 final 的,并且繼承自 Enum 類重绷。這些工作我們程序員沒做沸停,編譯器幫我們悄悄地做了。此外昭卓,它還附帶幾個(gè)有用靜態(tài)方法愤钾,比如說 values()valueOf(String name)

01候醒、內(nèi)部枚舉

好的能颁,小伙伴們應(yīng)該已經(jīng)清楚枚舉長(zhǎng)什么樣子了吧?既然枚舉是一種特殊的類倒淫,那它其實(shí)是可以定義在一個(gè)類的內(nèi)部的伙菊,這樣它的作用域就可以限定于這個(gè)外部類中使用。

public class Player {
    private PlayerType type;
    public enum PlayerType {
        TENNIS,
        FOOTBALL,
        BASKETBALL
    }
    
    public boolean isBasketballPlayer() {
      return getType() == PlayerType.BASKETBALL;
    }

    public PlayerType getType() {
        return type;
    }

    public void setType(PlayerType type) {
        this.type = type;
    }
}

PlayerType 就相當(dāng)于 Player 的內(nèi)部類敌土,isBasketballPlayer() 方法用來判斷運(yùn)動(dòng)員是否是一個(gè)籃球運(yùn)動(dòng)員镜硕。

由于枚舉是 final 的,可以確保在 Java 虛擬機(jī)中僅有一個(gè)常量對(duì)象(可以參照反編譯后的靜態(tài)代碼塊「static 關(guān)鍵字帶大括號(hào)的那部分代碼」)返干,所以我們可以很安全地使用“==”運(yùn)算符來比較兩個(gè)枚舉是否相等兴枯,參照 isBasketballPlayer() 方法。

那為什么不使用 equals() 方法判斷呢矩欠?

if(player.getType().equals(Player.PlayerType.BASKETBALL)){};
if(player.getType() == Player.PlayerType.BASKETBALL){};

“==”運(yùn)算符比較的時(shí)候念恍,如果兩個(gè)對(duì)象都為 null,并不會(huì)發(fā)生 NullPointerException晚顷,而 equals() 方法則會(huì)峰伙。

另外, “==”運(yùn)算符會(huì)在編譯時(shí)進(jìn)行檢查该默,如果兩側(cè)的類型不匹配瞳氓,會(huì)提示錯(cuò)誤,而 equals() 方法則不會(huì)栓袖。

02匣摘、枚舉可用于 switch 語句

這個(gè)我在之前的一篇我去的文章中詳細(xì)地說明過了,感興趣的小伙伴可以點(diǎn)擊鏈接跳轉(zhuǎn)過去看一下裹刮。

switch (playerType) {
        case TENNIS:
            return "網(wǎng)球運(yùn)動(dòng)員費(fèi)德勒";
        case FOOTBALL:
            return "足球運(yùn)動(dòng)員C羅";
        case BASKETBALL:
            return "籃球運(yùn)動(dòng)員詹姆斯";
        case UNKNOWN:
            throw new IllegalArgumentException("未知");
        default:
            throw new IllegalArgumentException(
                    "運(yùn)動(dòng)員類型: " + playerType);

    }

03音榜、枚舉可以有構(gòu)造方法

如果枚舉中需要包含更多信息的話,可以為其添加一些字段捧弃,比如下面示例中的 name赠叼,此時(shí)需要為枚舉添加一個(gè)帶參的構(gòu)造方法擦囊,這樣就可以在定義枚舉時(shí)添加對(duì)應(yīng)的名稱了。

public enum PlayerType {
    TENNIS("網(wǎng)球"),
    FOOTBALL("足球"),
    BASKETBALL("籃球");

    private String name;

    PlayerType(String name) {
        this.name = name;
    }
}

04嘴办、EnumSet

EnumSet 是一個(gè)專門針對(duì)枚舉類型的 Set 接口的實(shí)現(xiàn)類瞬场,它是處理枚舉類型數(shù)據(jù)的一把利器,非常高效(內(nèi)部實(shí)現(xiàn)是位向量涧郊,我也搞不懂)贯被。

因?yàn)?EnumSet 是一個(gè)抽象類,所以創(chuàng)建 EnumSet 時(shí)不能使用 new 關(guān)鍵字妆艘。不過彤灶,EnumSet 提供了很多有用的靜態(tài)工廠方法:

下面的示例中使用 noneOf() 創(chuàng)建了一個(gè)空的 PlayerType 的 EnumSet;使用 allOf() 創(chuàng)建了一個(gè)包含所有 PlayerType 的 EnumSet批旺。

public class EnumSetTest {
    public enum PlayerType {
        TENNIS,
        FOOTBALL,
        BASKETBALL
    }

    public static void main(String[] args) {
        EnumSet<PlayerType> enumSetNone = EnumSet.noneOf(PlayerType.class);
        System.out.println(enumSetNone);

        EnumSet<PlayerType> enumSetAll = EnumSet.allOf(PlayerType.class);
        System.out.println(enumSetAll);
    }
}

程序輸出結(jié)果如下所示:

[]
[TENNIS, FOOTBALL, BASKETBALL]

有了 EnumSet 后枢希,就可以使用 Set 的一些方法了:

05、EnumMap

EnumMap 是一個(gè)專門針對(duì)枚舉類型的 Map 接口的實(shí)現(xiàn)類朱沃,它可以將枚舉常量作為鍵來使用苞轿。EnumMap 的效率比 HashMap 還要高,可以直接通過數(shù)組下標(biāo)(枚舉的 ordinal 值)訪問到元素逗物。

和 EnumSet 不同搬卒,EnumMap 不是一個(gè)抽象類,所以創(chuàng)建 EnumMap 時(shí)可以使用 new 關(guān)鍵字:

EnumMap<PlayerType, String> enumMap = new EnumMap<>(PlayerType.class);

有了 EnumMap 對(duì)象后就可以使用 Map 的一些方法了:

和 HashMap 的使用方法大致相同翎卓,來看下面的例子:

EnumMap<PlayerType, String> enumMap = new EnumMap<>(PlayerType.class);
enumMap.put(PlayerType.BASKETBALL,"籃球運(yùn)動(dòng)員");
enumMap.put(PlayerType.FOOTBALL,"足球運(yùn)動(dòng)員");
enumMap.put(PlayerType.TENNIS,"網(wǎng)球運(yùn)動(dòng)員");
System.out.println(enumMap);

System.out.println(enumMap.get(PlayerType.BASKETBALL));
System.out.println(enumMap.containsKey(PlayerType.BASKETBALL));
System.out.println(enumMap.remove(PlayerType.BASKETBALL));

程序輸出結(jié)果如下所示:

{TENNIS=網(wǎng)球運(yùn)動(dòng)員, FOOTBALL=足球運(yùn)動(dòng)員, BASKETBALL=籃球運(yùn)動(dòng)員}
籃球運(yùn)動(dòng)員
true
籃球運(yùn)動(dòng)員

06契邀、單例

通常情況下,實(shí)現(xiàn)一個(gè)單例并非易事失暴,不信坯门,來看下面這段代碼

public class Singleton {  
    private volatile static Singleton singleton; 
    private Singleton (){}  
    public static Singleton getSingleton() {  
    if (singleton == null) {
        synchronized (Singleton.class) { 
        if (singleton == null) {  
            singleton = new Singleton(); 
        }  
        }  
    }  
    return singleton;  
    }  
}

但枚舉的出現(xiàn),讓代碼量減少到極致:

public enum EasySingleton{
    INSTANCE;
}

完事了逗扒,真的超級(jí)短古戴,有沒有?枚舉默認(rèn)實(shí)現(xiàn)了 Serializable 接口矩肩,因此 Java 虛擬機(jī)可以保證該類為單例现恼,這與傳統(tǒng)的實(shí)現(xiàn)方式不大相同。傳統(tǒng)方式中黍檩,我們必須確保單例在反序列化期間不能創(chuàng)建任何新實(shí)例叉袍。

07、枚舉可與數(shù)據(jù)庫(kù)交互

我們可以配合 Mybatis 將數(shù)據(jù)庫(kù)字段轉(zhuǎn)換為枚舉類型」艚矗現(xiàn)在假設(shè)有一個(gè)數(shù)據(jù)庫(kù)字段 check_type 的類型如下:

`check_type` int(1) DEFAULT NULL COMMENT '檢查類型(1:未通過喳逛、2:通過)',

它對(duì)應(yīng)的枚舉類型為 CheckType,代碼如下:

public enum CheckType {
    NO_PASS(0, "未通過"), PASS(1, "通過");
    private int key;

    private String text;

    private CheckType(int key, String text) {
        this.key = key;
        this.text = text;
    }

    public int getKey() {
        return key;
    }

    public String getText() {
        return text;
    }

    private static HashMap<Integer,CheckType> map = new HashMap<Integer,CheckType>();
    static {
        for(CheckType d : CheckType.values()){
            map.put(d.key, d);
        }
    }
    
    public static CheckType parse(Integer index) {
        if(map.containsKey(index)){
            return map.get(index);
        }
        return null;
    }
}

1)CheckType 添加了構(gòu)造方法棵里,還有兩個(gè)字段润文,key 為 int 型姐呐,text 為 String 型。

2)CheckType 中有一個(gè)public static CheckType parse(Integer index)方法转唉,可將一個(gè) Integer 通過 key 的匹配轉(zhuǎn)化為枚舉類型皮钠。

那么現(xiàn)在稳捆,我們可以在 Mybatis 的配置文件中使用 typeHandler 將數(shù)據(jù)庫(kù)字段轉(zhuǎn)化為枚舉類型赠法。

<resultMap id="CheckLog" type="com.entity.CheckLog">
  <id property="id" column="id"/>
  <result property="checkType" column="check_type" typeHandler="com.CheckTypeHandler"></result>
</resultMap>

其中 checkType 字段對(duì)應(yīng)的類如下:

public class CheckLog implements Serializable {

    private String id;
    private CheckType checkType;

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public CheckType getCheckType() {
        return checkType;
    }

    public void setCheckType(CheckType checkType) {
        this.checkType = checkType;
    }
}

CheckTypeHandler 轉(zhuǎn)換器的類源碼如下:

public class CheckTypeHandler extends BaseTypeHandler<CheckType> {

    @Override
    public CheckType getNullableResult(ResultSet rs, String index) throws SQLException {
        return CheckType.parse(rs.getInt(index));
    }

    @Override
    public CheckType getNullableResult(ResultSet rs, int index) throws SQLException {
        return CheckType.parse(rs.getInt(index));
    }

    @Override
    public CheckType getNullableResult(CallableStatement cs, int index) throws SQLException {
        return CheckType.parse(cs.getInt(index));
    }

    @Override
    public void setNonNullParameter(PreparedStatement ps, int index, CheckType val, JdbcType arg3) throws SQLException {
        ps.setInt(index, val.getKey());
    }
}

CheckTypeHandler 的核心功能就是調(diào)用 CheckType 枚舉類的 parse() 方法對(duì)數(shù)據(jù)庫(kù)字段進(jìn)行轉(zhuǎn)換。

恕我直言乔夯,這篇文章看完后砖织,我覺得小伙伴們肯定會(huì)用 Java 枚舉了,如果還不會(huì)末荐,就過來砍我侧纯!

如果覺得文章對(duì)你有點(diǎn)幫助,請(qǐng)微信搜索「 沉默王二 」第一時(shí)間閱讀甲脏,回復(fù)「并發(fā)」更有一份阿里大牛重寫的 Java 并發(fā)編程實(shí)戰(zhàn)眶熬,從此再也不用擔(dān)心面試官在這方面的刁難了。

本文已收錄 GitHub块请,傳送門~ 娜氏,里面更有大廠面試完整考點(diǎn),歡迎 Star墩新。

我是沉默王二贸弥,一枚有顏值卻靠才華茍且的程序員。關(guān)注即可提升學(xué)習(xí)效率海渊,別忘了三連啊绵疲,點(diǎn)贊、收藏臣疑、留言盔憨,我不挑,嘻嘻讯沈。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末般渡,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子芙盘,更是在濱河造成了極大的恐慌驯用,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,839評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件儒老,死亡現(xiàn)場(chǎng)離奇詭異蝴乔,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)驮樊,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門薇正,熙熙樓的掌柜王于貴愁眉苦臉地迎上來片酝,“玉大人,你說我怎么就攤上這事挖腰〉裱兀” “怎么了?”我有些...
    開封第一講書人閱讀 153,116評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵猴仑,是天一觀的道長(zhǎng)审轮。 經(jīng)常有香客問我,道長(zhǎng)辽俗,這世上最難降的妖魔是什么疾渣? 我笑而不...
    開封第一講書人閱讀 55,371評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮崖飘,結(jié)果婚禮上榴捡,老公的妹妹穿的比我還像新娘。我一直安慰自己朱浴,他們只是感情好吊圾,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,384評(píng)論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著翰蠢,像睡著了一般项乒。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上躏筏,一...
    開封第一講書人閱讀 49,111評(píng)論 1 285
  • 那天板丽,我揣著相機(jī)與錄音,去河邊找鬼趁尼。 笑死埃碱,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的酥泞。 我是一名探鬼主播砚殿,決...
    沈念sama閱讀 38,416評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼芝囤!你這毒婦竟也來了似炎?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,053評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤悯姊,失蹤者是張志新(化名)和其女友劉穎羡藐,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體悯许,經(jīng)...
    沈念sama閱讀 43,558評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡仆嗦,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,007評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了先壕。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片瘩扼。...
    茶點(diǎn)故事閱讀 38,117評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡谆甜,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出集绰,到底是詐尸還是另有隱情规辱,我是刑警寧澤,帶...
    沈念sama閱讀 33,756評(píng)論 4 324
  • 正文 年R本政府宣布栽燕,位于F島的核電站罕袋,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏纫谅。R本人自食惡果不足惜炫贤,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,324評(píng)論 3 307
  • 文/蒙蒙 一溅固、第九天 我趴在偏房一處隱蔽的房頂上張望付秕。 院中可真熱鬧,春花似錦侍郭、人聲如沸询吴。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,315評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽猛计。三九已至,卻和暖如春爆捞,著一層夾襖步出監(jiān)牢的瞬間奉瘤,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,539評(píng)論 1 262
  • 我被黑心中介騙來泰國(guó)打工煮甥, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留盗温,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,578評(píng)論 2 355
  • 正文 我出身青樓成肘,卻偏偏與公主長(zhǎng)得像卖局,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子双霍,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,877評(píng)論 2 345