Enum 类型的介绍
枚举类型是Java5新增的特性之一,枚举是一种特殊类型的类,其枚举的每一个值都是该枚举类的一个实例。枚举类型是作为Java语言的一部分,是完全类型安全的,编译器会帮助我们检查枚举类型的正确性。
在枚举类型出现之前,当需要定义一些常量时,我们经常使用接口的静态常量来实现。这种方式没有错误,但没有枚举类型安全,将接口中的不同静态常量初始化为相同值时,编译并不会发出警告,所以我们更提倡使用枚举类型
接口方式实现
interface Color {
int RED = 0;
int YELLOW = 1;
int BLUE = 2;
}
//使用
int yellow = Color.YELLOW;
System.out.println(yellow); //1
知识扩展:interface 中可以定义成员变量的,但是定义的成员变量都是属于常量,如下面两种方式都是等价的。
int RED = 0;
//等价于
public final int RED = 0;
枚举类的方式实现
enum Color {
RED,YELLOW, BLUE;
}
//使用
System.out.println(Color.YELLOW); //1
枚举声明是由关键字enum、该枚举类型的标识符及枚举常量构成的,这些常量是枚举类的静态字段,每个常量实际上是该枚举类的一个实例,常量之间用逗号隔开。枚举类中除了枚举常量也可以声明其他成员,但在声明其他成员之前,必须先声明所有的枚举常量,而且最后一个常量以分号结尾。
enum Color {
//当有其他成员时,最后一个常量以分号结尾
RED("red"),YELLOW("yellow"), BLUE("blue");
private String value;
//私有构造函数
private Color(String value) {
this.value = value;
}
public String getValue() {
return value;
}
}
//使用
System.out.println(Color.RED.getValue());//red
枚举常量都是在类内声明的,为防止在外部声明,构造函数需要用private修饰,可省略不写,而且在构造器中不能使用该枚举类的非静态常量字段。声明枚举常量时,不可以用new来构建实例,而是像上面代码那样,直接用枚举常量来声明,有参数时将参数写在一对括号之间。
枚举的原理
在使用enum创建枚举类型并编译后,编译器会将枚举类编译成独立的类文件,为了深入地理解枚举类的构建原理,我用Java Decompiler将枚举类进行了反编译,代码如下:
//反编译后的代码
final class Color extends Enum
{
public static Color[] values()
{
return (Color[])$VALUES.clone();
}
public static Color valueOf(String s)
{
return (Color)Enum.valueOf(Color, s);
}
private Color(String s, int i)
{
super(s, i);
}
public static final Color RED;
public static final Color YELLOW;
public static final Color BLUE;
private static final Color $VALUES[];
static
{
RED = new Color("RED", 0);
YELLOW = new Color("YELLOW", 1);
BLUE = new Color("BLUE", 2);
$VALUES = (new Color[] {
RED, YELLOW, BLUE
});
}
}
从反编译后的代码中,我们可以看出以下几点:
- 枚举类继承了Enum类。
- 枚举类隐式地继承了java.lang.Enum,Java类只支持单继承,所以枚举类不能继承其他类,但可以实现接口。
- 枚举类是final的,我们不能再继承枚举类。
- 枚举常量通过静态块来进行初始化,序数是从0开始的。
- 编译器为我们自动增加了values()和valueof(String name)方法。
public static E[ ] values():返回包含了所有枚举常量的数组。
public static E valueof(String name): 返回指定名字的枚举常量。
结束
枚举类作为Java语言的一部分,是类型安全的,并且在switch语句中也可以使用枚举类型,这为我们提供了很多方便。当我们需要声明一个有关实例的人所共知的闭集时,枚举类型是一个不错的选择。