JDK源码解析之 Java.lang.Enum
Enum是一個(gè)特殊的類(lèi). 我們不能以class Xxx extends Enum的方式手動(dòng)繼承, 必須寫(xiě)成enum Xxx的形式; 然而這段枚舉類(lèi)的定義在編譯之后又變回了class Xxx extends Enum.
一、類(lèi)定義
public abstract class Enum<E extends Enum<E>>implements Comparable<E>, Serializable {}- Enum是一個(gè)虛基類(lèi),
- 繼承了Number抽象類(lèi),可以用于數(shù)字類(lèi)型的一系列轉(zhuǎn)換;
- 實(shí)現(xiàn)了Comparable接口,強(qiáng)行對(duì)實(shí)現(xiàn)它的每個(gè)類(lèi)的對(duì)象進(jìn)行整體排序
二、成員常量和取值器
//成員常量name,被聲明為final類(lèi)型,意為不可變,同時(shí)只提供取值器。 private final String name; public final String name() {return name; } //成員常量ordinal,被聲明為final類(lèi)型,意為不可變,同時(shí)只提供取值器。 private final int ordinal; public final int ordinal() {return ordinal; }三、構(gòu)造函數(shù)
protected Enum(String name, int ordinal) {this.name = name;this.ordinal = ordinal;}構(gòu)造函數(shù)被定義成protected,參數(shù)有兩個(gè),分別是name和ordinal,和成員常量相呼應(yīng)。
四、常用方法
1、toString()
返回成員常量name的值。
public String toString() {return name;}2、equals()
參數(shù)為Object類(lèi)型,方法內(nèi)部判斷是不是和this指向同一對(duì)象。注意這個(gè)方法final被修飾,也就是說(shuō)枚舉不可以覆寫(xiě)這個(gè)類(lèi)。
public final boolean equals(Object other) {return this==other;}3、hashCode()
委托父類(lèi)執(zhí)行,即Object類(lèi)的hashCode方法執(zhí)行。注意這個(gè)方法final被修飾,也就是說(shuō)枚舉不可以覆寫(xiě)這個(gè)類(lèi)。
public final int hashCode() {return super.hashCode();}4、clone()
直接拋出異常,不支持拷貝。這也符合枚舉唯一的特性。值得注意的是,這個(gè)方法被定義成protected,也就是只能包內(nèi)和子類(lèi)可見(jiàn),并且不可以覆寫(xiě)。
protected final Object clone() throws CloneNotSupportedException {throw new CloneNotSupportedException();}5、compareTo()
有這個(gè)方法,一方面是因?yàn)镋num類(lèi)實(shí)現(xiàn)了Comparable接口,一方面也符合枚舉可以比較的特點(diǎn)。值得注意的是,方法內(nèi)部可以看出,要求比較的對(duì)象必須是相同枚舉類(lèi)型的,否則便會(huì)引起ClassCastException異常。對(duì)相同類(lèi)型枚舉值,比較的則是它們的成員常量ordinal大小。
public final int compareTo(E o) {Enum<?> other = (Enum<?>)o;Enum<E> self = this;if (self.getClass() != other.getClass() && // optimizationself.getDeclaringClass() != other.getDeclaringClass())throw new ClassCastException();return self.ordinal - other.ordinal;}6、getDeclaringClass()
這個(gè)方法在compareTo()方法中出現(xiàn)了。
@SuppressWarnings("unchecked")public final Class<E> getDeclaringClass() {Class<?> clazz = getClass();Class<?> zuper = clazz.getSuperclass();return (zuper == Enum.class) ? (Class<E>)clazz : (Class<E>)zuper;}7、valueOf()
通過(guò)參數(shù)name來(lái)返回枚舉值,注意,如果name沒(méi)有與之對(duì)應(yīng)的枚舉,返回的不是null,而是拋出異常。
public static <T extends Enum<T>> T valueOf(Class<T> enumType,String name) {T result = enumType.enumConstantDirectory().get(name);if (result != null)return result;if (name == null)throw new NullPointerException("Name is null");throw new IllegalArgumentException("No enum constant " + enumType.getCanonicalName() + "." + name);}8、readObject()
防治反序列化的,使用private來(lái)定義方法,且在方法內(nèi)部拋出異常。目的就是為了維護(hù)枚舉值的唯一性。
private void readObject(ObjectInputStream in) throws IOException,ClassNotFoundException {throw new InvalidObjectException("can't deserialize enum");}9、readObjectNoData()
同readObject方法一樣,都是為了防止反序列化的。
private void readObjectNoData() throws ObjectStreamException {throw new InvalidObjectException("can't deserialize enum");}總結(jié)
以上是生活随笔為你收集整理的JDK源码解析之 Java.lang.Enum的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 编写第一个Spring程序——IOC实现
- 下一篇: JDK源码解析之 Java.lang.S