枚舉(Enum)類型是 java 5 中新增的一種數(shù)據(jù)類型,它能夠幫助我們更加快捷和安全的實(shí)現(xiàn)枚舉。
回想之前我們?cè)诙x枚舉常量時(shí)的做法:
這樣的定義方式雖然也能正常工作,但卻存在許多不足,比如不小心把 MONDAY 和 TUESDAY 都置為 2 時(shí),編譯器并不會(huì)報(bào)錯(cuò),但是卻很難進(jìn)行排查。
在 Java 5 之后,我們可以用如下的定義定義枚舉類型。并可以在程序中通過(guò) 的方式使用,這無(wú)疑大大提高了程序的安全性。
我們知道,編程語(yǔ)言的設(shè)計(jì)具有前向兼容性,這意味著后續(xù)的語(yǔ)言特性實(shí)際上都是通過(guò)語(yǔ)法糖來(lái)實(shí)現(xiàn)的,那么枚舉類型的內(nèi)部實(shí)現(xiàn)原理是怎么樣的呢?我們通過(guò)將上面的枚舉類 Day 通過(guò) 命令進(jìn)行反編譯,再將反編譯的代碼通過(guò)改寫(xiě)使其更加容易閱讀,最終得到下面的反編譯代碼。
通過(guò)閱讀上面的反編譯后代碼,我們可以對(duì)枚舉類型的實(shí)現(xiàn)進(jìn)行總結(jié):
枚舉類型是通過(guò)繼承 Enum 類來(lái)實(shí)現(xiàn)的,并且最終生成 final 類來(lái)強(qiáng)化不可變性。
枚舉類型的構(gòu)造函數(shù)為私有的,具有 String 和 int 兩個(gè)參數(shù),分別代表枚舉的名稱和序號(hào),序號(hào)按照定義的順序從小到大排列。
每一個(gè)枚舉都代表一個(gè)枚舉類事例,并且為 static 和 final,在枚舉類的靜態(tài)代碼塊中進(jìn)行初始化,并且有一個(gè) $VALUES 數(shù)組保存所有的枚舉。
通過(guò) valueOf 方法可以完成枚舉名稱到枚舉對(duì)象的查找。
VALUES()方法返回的是 VALUES 數(shù)組被篡改。
接下來(lái),我們來(lái)分析一下自定義枚舉類型所繼承的 Enum 類
通過(guò)閱讀上面的 Enum 抽象類,我們可以對(duì)其特點(diǎn)進(jìn)行總結(jié):
由 equals 方法可知每個(gè)枚舉只與自身相等,沒(méi)有等效相等的枚舉。
由 clone 方法可知枚舉對(duì)象不允許克隆,這能保證每一個(gè)枚舉都是唯一的。
由 compareTo 方法可知枚舉只能與同類型的枚舉相比較,返回結(jié)果為枚舉的順序之差。
由 finalize 方法可知枚舉類不允許實(shí)現(xiàn) finalize 方法,這與枚舉的安全性有關(guān)。
由 readObject 方法和 readObjectNoData 方法可知枚舉對(duì)象不允許反序列化,這也能保證每一個(gè)枚舉都是唯一的。
由以上的分析我們可以發(fā)現(xiàn),枚舉的最大特點(diǎn)就是唯一性,同時(shí)可以發(fā)現(xiàn)只有單個(gè)元素的枚舉在不經(jīng)意間符合了單例模式的要求,具體為:
枚舉類為不可變類,這防止了單例類被繼承。
枚舉類型的構(gòu)造函數(shù)為私有的,因此不能主動(dòng)創(chuàng)建對(duì)應(yīng)的單例對(duì)象。
枚舉的元素為 public static final 類型,并且在類加載的時(shí)候在靜態(tài)代碼塊內(nèi)完成了初始化,這相當(dāng)于單例模式的惡漢模式。
枚舉類對(duì)象不允許克隆,這能保證每一個(gè)單例都是唯一的。
枚舉類對(duì)象不可反序列化,這也能保證每一個(gè)單例都是唯一的。
因此我們可以使用單元素的枚舉類型來(lái)實(shí)現(xiàn)單例模式。事實(shí)上,正如Effective Java一書(shū)中所說(shuō)的:?jiǎn)卧氐拿杜e類型已經(jīng)成為實(shí)現(xiàn) Singleton 的最佳方法。
注意:值得提出的時(shí),由以上的分析我們知道每一個(gè)枚舉都是一個(gè)對(duì)象,既然是對(duì)象,那么它所占的內(nèi)存就比基本類型大很多,這就是枚舉類型的缺點(diǎn),所以在Android開(kāi)發(fā)中并不建議使用枚舉類型,而是使用等注解加上int或者String進(jìn)行替代。
咨詢熱線
010-85377344
135-21581588
微信客服
QQ客服
3026106565 點(diǎn)擊咨詢