枚举类的存在意义
在某些情况下,一个类的对象是有限而且固定的。比如季节类,它只有四个对象;再比如行星类,它只有八个对象。这种实例有限且固定的类,在Java里被叫做枚举类。
在早期代码中,会直接使用简单的静态常量来表示枚举(例如public static final int SEASON_SPRING = 1;),但存在如下几个问题:1.类型不安全(可以进行本毫无意义的运算且不会报错)2.没有命名空间(所以名称必须唯一,否则会报错)3.打印输出的意义不明(会打印出来int值)。所以枚举类就有了存在的必要。
枚举类
enum关键字(它与class、interface关键字地位相同)用以定义枚举类。枚举类是一种特殊的类,他一样可以有自己的成员变量、方法,可以实现一个或者多个接口,也可以定义自己的构造器。一个Java源文件中最多只能定义一个public访问权限的枚举类,且该Java源文件也必须和该枚举类的类名相同。
「在一个Java源文件当中可以有多个类,但只能有一个public类,而当这个类被修饰为public的话,源文件名必须要与此类名相同。一个Java源文件中包含多个类时,此时只能够有一个public类,因为Java程序的入口是main方法,所以被定为public的这个类里一定是含有main方法的类,而且该类的名称要和文件名一致,因为虚拟机开始要找main方法的。」
但枚举类终究不是普通类,它与普通类有如下简单区别:1.枚举类默认继承了java. lang.Enum类(java.lang.Enum类中包含多种方法,具体查阅文档),而不是默认继承Object类,因此枚举类不能显式继承其他父类,其中java.lang.Enum类实现了Serializable和Comparable两个接口。2.非抽象的枚举类默认会使用final修饰,因此枚举类不能派生子类。3.枚举类的构造器只能使用private访问控制符,如果省略了构造器的访问控制符,则默认使用private修饰,如果强制指定访问控制符,则只能指定为private。4.枚举类的所有实例必须在枚举类的第一行显式列出(如例),否则这个枚举类永远都不能产生实例。列出这些实例时,系统会自动添加public static final修饰,无需显式添加。
枚举类默认提供了一个values方法,该方法可以很方便地遍历所有的枚举值。编译上面Java程序,将生成一个SeasonEnum.class文件,这表明枚举类是一个特殊的Java类。如果需要使用该枚举类的某个实例,则可使用Enumclass.variable的形式,如SeasonEnum.SPRING或者使用Enum类的静态方法来获取实例。枚举类的实例只能是枚举值,而不是随意地通过new来创建枚举类对象。
枚举类的成员变量、方法和构造器
枚举类通常应该设计成不可变类,也就是说如果这个枚举类拥有成员变量的话,那么它的成员变量值不应该允许改变。因此建议将枚举类的成员变量都使用private final修饰。如果将所有的成员变量都使用了final修饰符来修饰,那么必须在构造器里为这些成员变量指定初始值,因此应该为枚举类显式定义带参数的构造器。一旦为枚举类显式定义了带参数的构造器,列出枚举值时就必须对应地传入参数。
在枚举类中列出枚举值时,实际上就是调用构造器创建枚举类对象,只是这里无须使用new关键字也无须显式调用构造器。前面列出枚举值时无须传入参数,甚至无须使用括号,仅仅是因为前面的枚举类包含无参数的构造器。不难看出,上面程序中第一行代码实际上等同于如下两行代码:public static final Gender MALE = new Gender("男");public static final Gender FEMALE new Gender("女");
实现接口的枚举类
「接口的方法默认public abstract。」
枚举类实现接口与普通类实现接口完全一样。如果由枚举类来实现接口里的方法,则每个枚举值在调用该方法时都有相同的行为方式(因为方法体完全一样)。如果需要每个枚举值在调用该方法时呈现出不同的行为方式,则可以让每个枚举值分别来实现该方法,每个枚举值提供不同的实现方式,从而让不同的枚举值调用该方法时具有不同的行为方式。
此时当创建MALE和FEMALE枚举值时,并不是直接创建Gender枚举类的实例,而是相当于创建Gender枚举类的匿名子类的实例。花括号部分实际上是匿名内部类的类体部分。
非抽象枚举类默认使用final修饰(即不可以派生子类),但抽象枚举类(因为它实现了接口所以包含了接口的抽象方法但并没有实现抽象方法,所以是抽象枚举类)可以派生子类。
「一个类实现了一个或多个接口之后,这个类必须完全实现这些接口里定义的全部抽象方法,否则该类将保留从父接口那里继承到的抽象方法,该类也必须定义成抽象类。」
编译上面的程序,可以看到生成了Gender.class、Gender$1.class和Gender$2.class三个文件,这样的三个class文件正好证明了上面的结论:MALE和FEMALE实际上是Gender匿名子类的实例,而不是Gender类的实例。当调用MALE和FEMALE两个枚举值的方法时,就会看到两个枚举值的方法表现不同的行为方式。
包含抽象方法的枚举类
即可以像上面所提的实现接口从而使枚举类变为抽象枚举类,进而实现不同枚举值的方法具有不同的行为方式,也可以直接在枚举类中定义抽象方法。
编译上面程序会生成5个cass文件,其实Operation对应一个class文件,它的4个匿名内部子类分别各对应一个cass文件。枚举类里定义抽象方法时不能使用abstract关键字将枚举类定义成抽象类(因为系统自动会为它添加abstract关键字),但因为枚举类需要显式创建枚举值,而不是作为父类,所以定义每个枚举值时必须为抽象方法提供实现,否则将出现编译错误。