Enum是一個特殊的類. 我們不能以class Xxx extends Enum
的方式手動繼承, 必須寫成enum Xxx
的形式; 然而這段枚舉類的定義在編譯之后又變回了class Xxx extends Enum
.
一、類定義
public abstract class Enum<E extends Enum<E>>implements Comparable<E>, Serializable {}
- Enum是一個虛基類,
- 繼承了Number抽象類,可以用于數字類型的一系列轉換;
- 實現了Comparable接口,強行對實現它的每個類的對象進行整體排序
二、成員常量和取值器
//成員常量name,被聲明為final類型,意為不可變,同時只提供取值器。
private final String name;
public final String name() {return name;
}
//成員常量ordinal,被聲明為final類型,意為不可變,同時只提供取值器。
private final int ordinal;
public final int ordinal() {return ordinal;
}
三、構造函數
protected Enum(String name, int ordinal) {this.name = name;this.ordinal = ordinal;}
構造函數被定義成protected,參數有兩個,分別是name和ordinal,和成員常量相呼應。
四、常用方法
1、toString()
返回成員常量name的值。
public String toString() {return name;}
2、equals()
參數為Object類型,方法內部判斷是不是和this指向同一對象。注意這個方法final被修飾,也就是說枚舉不可以覆寫這個類。
public final boolean equals(Object other) {return this==other;}
3、hashCode()
委托父類執行,即Object類的hashCode方法執行。注意這個方法final被修飾,也就是說枚舉不可以覆寫這個類。
public final int hashCode() {return super.hashCode();}
4、clone()
直接拋出異常,不支持拷貝。這也符合枚舉唯一的特性。值得注意的是,這個方法被定義成protected,也就是只能包內和子類可見,并且不可以覆寫。
protected final Object clone() throws CloneNotSupportedException {throw new CloneNotSupportedException();}
5、compareTo()
有這個方法,一方面是因為Enum類實現了Comparable接口,一方面也符合枚舉可以比較的特點。值得注意的是,方法內部可以看出,要求比較的對象必須是相同枚舉類型的,否則便會引起ClassCastException異常。對相同類型枚舉值,比較的則是它們的成員常量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()
這個方法在compareTo()方法中出現了。
@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()
通過參數name來返回枚舉值,注意,如果name沒有與之對應的枚舉,返回的不是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來定義方法,且在方法內部拋出異常。目的就是為了維護枚舉值的唯一性。
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");}