java java.lang.enum_源码阅读-java基础-java.lang.Enum

1、引言

枚举类型是 JDK 5 之后引进的一种非常重要的引用类型,可以用来定义一系列枚举常量。相比与常量(public static final定义),在安全性、指意性、可读性方面更胜一筹。另外它可以和switch case搭配使用。

2、类定义

实际上在使用关键字enum创建枚举类型并编译后,编译器会为我们生成一个相关的类,这个类继承了Java API中的java.lang.Enum类,也就是说通过关键字enum创建枚举类型在编译后事实上也是一个类类型而且该类继承自java.lang.Enum类,并且该类类型是final修饰的,无法被继承。

public abstract class Enum>implements Comparable, Serializable {}

3、成员属性

// 枚举常量的名称

private final String name;

// 它和不重写的toString()方法返回的是一样的,只不过toString()方法可以进行重写定制。

public final String name() {

return name;

}

/**

* 枚举常量的序号(它在枚举声明中的位置,其中初始常量的序号为零)。

* 代码中尽量避免使用该值,因为我们有时候添加新枚举时,总是“按组”放在一起,不会放到最后,除非强制要求。

* 比如接口响应码的枚举,我们通常会把参数异常的放一组,内部异常的放一组等。如果参数异常新添加一个枚举“xx不存在”,如果这个枚举没有放在整个接口响应码枚举的末尾,而是放到了某一组的末尾,那么部分枚举常量的ordinal值就发生了变化。这将是灾难性的,所以要么代码规范添加新枚举只能添加到末尾;要么不使用该值。

*/

private final int ordinal;

public final int ordinal() {

return ordinal;

}

4、构造方法

// 唯一的构造方法,程序员调用不到,它由编译器发出的代码用于响应枚举类型声明。

protected Enum(String name, int ordinal) {

this.name = name;

this.ordinal = ordinal;

}

5、其他方法

// 返回声明中包含的此枚举常量的名称。 可以覆盖该方法,但通常不需要或不需要。

public String toString() {

return name;

}

public final boolean equals(Object other) {

return this==other;

}

// 这里调用的是Object的hashCode方法

public final int hashCode() {

return super.hashCode();

}

// 不能克隆,保证了枚举永远是单例的,所以枚举是单例模式的典型代表

protected final Object clone() throws CloneNotSupportedException {

throw new CloneNotSupportedException();

}

// 这里需要注意,底层的实现用的是ordinal实现的,即比较的是枚举常量的序号。

public final int compareTo(E o) {

Enum> other = (Enum>)o;

Enum self = this;

if (self.getClass() != other.getClass() &&

self.getDeclaringClass() != other.getDeclaringClass())

throw new ClassCastException();

return self.ordinal - other.ordinal;

}

// 返回与此枚举常量的枚举类型相对应的 Class 对象。

public final Class getDeclaringClass() {

Class> clazz = getClass();

Class> zuper = clazz.getSuperclass();

return (zuper == Enum.class) ? (Class)clazz : (Class)zuper;

}

// 枚举不能有finalize方法

protected final void finalize() { }

/**

* 防止默认反序列化

*/

private void readObject(ObjectInputStream in) throws IOException,ClassNotFoundException {

throw new InvalidObjectException("can't deserialize enum");

}

private void readObjectNoData() throws ObjectStreamException {

throw new InvalidObjectException("can't deserialize enum");

}

/**

* 返回简单名称到枚举常量的映射。

* name的值必须与用于声明此类型的枚举常量的标识符完全一致。

*

* 对于特定的枚举类型T,有两个隐式声明的方法可以直接使用:

* 1) public static T valueOf(String) 根据名称获取单个枚举类型;这个我深有体会,因为之前是不知道有这个方法,通常是用values()获取枚举集合map,然后遍历

* 2) public static T[] values() 获取所有枚举类型数组;

*/

public static > T valueOf(Class 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);

}

/**

* 1)该方法是属于java.lang.Class,不是java.lang.Enum的方法;

* 2)enumConstantDirectory 是一个map,定义如下:private volatile transient Map enumConstantDirectory = null;其中key为枚举的name,value为枚举。

* 3)enumConstantDirectory()方法获取枚举常量目录,没有就继续调用getEnumConstantsShared();如果有返回值,就遍历put进enumConstantDirectory。

*/

Map enumConstantDirectory() {

if (enumConstantDirectory == null) {

T[] universe = getEnumConstantsShared();

if (universe == null)

throw new IllegalArgumentException(

getName() + " is not an enum type");

Map m = new HashMap<>(2 * universe.length);

for (T constant : universe)

m.put(((Enum>)constant).name(), constant);

enumConstantDirectory = m;

}

return enumConstantDirectory;

}

7、枚举隐式实现的方法

如上所说,枚举自动继承这两个隐式方法。这两个方法很重要,所以我在这里单拎出来了。

// 根据名称获取单个枚举类型

public static T valueOf(String){}

// 获取所有枚举类型数组

public static T[] values(){}

8、枚举集合

这两个是枚举优化过后的集合,比HashSet,HashMap的性能更高。

8.1、EnumSet

8.2、EnumMap

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值