电脑故障

位置:IT落伍者 >> 电脑故障 >> 浏览文章

驯服Tiger之深入研究枚举类型


发布日期:2023/10/8
 

基础知识

在JSE中通过使用新的 enum 关键字创建指定的对象集合您可以创建一个枚举类型然后可以将每个指定的值看作是那个类的一个实例这为您提供了指定的整数集合所无法提供的编译时类型安全清单 将创建一个枚举类型并将类型安全的枚举值作为帮助器方法(helper method)的参数该枚举类型的 values() 方法返回这种类型的不同值的有序数组

清单 枚举类型的例子

public class Loop {

enum Size {

Small

Medium

Large

}

public static void main(String args[]) {

for (Size s : Sizevalues()) {

helper(s);

}

}

private static void helper(Size s) {

Systemoutprintln(Size value: + s);

}

}

构造函数方法和变量

在使用 enum 关键字创建新的枚举类型时实际上是在创建 javalangEnum 类的子类其中枚举类型符合通用模式 Class Enum<E extends Enum<E>>而 E 表示枚举类型的名称枚举类型的每一个值都将映射到 protected Enum(String name int ordinal) 构造函数中在这里每个值的名称都被转换成一个字符串并且序数设置表示了每个设置的优先值换句话说enum Size {Small Medium Large} 将映射到清单 中所示的构造函数调用中

清单 映射的构造函数调用

new Enum<Size>(Small );

new Enum<Size>(Medium );

new Enum<Size>(Large );

不必将构造函数的使用限制为间接 Enum 构造函数调用在使用 enum 关键字时将创建 Enum 的子类您可以使用参数和任何别的东西为定义的每个名称添加一些您自己的构造函数调用名称声明可以看作是对构造函数的调用您不必添加 new 关键字这种方法允许您将数据作为参数值传递给构造函数调用如清单 所示该参数表示 Size 对象的枚举集合的定价因子位于枚举类型定义之后的 main() 方法演示了这种用法

清单 定制构造函数的例子

public class Sample {

enum Size {

Small()

Medium()

Large();

double pricingFactor;

Size(double p) {

pricingFactor = p;

}

}

public static void main(String args[]) {

Size s = SizeLarge;

double d = spricingFactor;

Systemoutprintln(s + Size has pricing factor of + d);

}

}

运行该程序将返回给定 Size 的定价因子您还可以定义一个类似于 getPricingFactor() 的方法并将 pricingFactor 字段设置为 private以便更多地将它作为类 JavaBean 的属性对待清单 给前面的例子添加了一个方法

清单 方法的例子

public class Sample {

enum Size {

Small()

Medium()

Large();

private double pricingFactor;

Size(double p) {

pricingFactor = p;

}

public double getPricingFactor() {

return pricingFactor;

}

}

public static void main(String args[]) {

Size s = SizeLarge;

double d = sgetPricingFactor();

Systemoutprintln(s + Size has pricing factor of + d);

}

}

对于这两种情况输出均为

Large Size has pricing factor of

预定义的方法

因为用户定义的枚举类型是 Enum 类型的子类所以您需要继承用于您的类型的那个类的所有方法下面列出了完整的方法集合(E 表示枚举类型自身)

public int compareTo(E e)

public boolean equals(Object o)

public final Class<E> getDeclaringClass()

public int hashCode()

public String name()

public int ordinal()

public String toString()

public static <T extends Enum<T>> T valueOf(Class<T> enumType String name)

一些方法看起来很熟悉而其他一些方法则是特定于 Enum 类的compareTo()equals() 和 hashCode() 方法是典型的 Object 和 Comparable 方法其中compareTo() 报告声明元素的顺序name() 和 ordinal() 方法返回构造函数参数而 toString() 返回名称

getDeclaringClass() 和 valueOf() 方法需要稍多一些解释getDeclaringClass() 方法类似于 Object 的 getClass() 方法但它没必要返回相同的类根据这个方法的 Javadoc 的说明

对于具有特定于常量的类主体的 enum 常量该方法返回的值可能不同于 ObjectgetClass() 方法返回的值

接下来我将解释特定于常量的类主体valueOf() 方法是静态的它允许您从类型的名称中创建枚举的值

特定于常量的类主体

特定于常量的类主体是 enum 关键字的一个受支持的特性不过它们的使用应该受到严格的限制这个概念正在深入到将枚举类型的每个元素作为一个子类对待的领域例如在前面的例子中Size 枚举类型有一个定价因子参数和 getPricingFactor() 方法但没有构造函数参数清单 展示了如何利用特定于常量的主体来做同样的事我们添加了一些额外的大小来让这个例子更有趣些在这里Small 的定价因子是 而 ExtraLarge 和 ExtraExtraLarge 的定价因子是 其余的大小则采用默认值

清单 特定于常量的主体

public class Sample {

enum Size {

Small {

public double getPricingFactor() {

return ;

}

}

Medium

Large

ExtraLarge {

public double getPricingFactor() {

return ;

}

}

ExtraExtraLarge {

public double getPricingFactor() {

return ;

}

};

public double getPricingFactor() {

return ;

}

}

public static void main(String args[]) {

for (Size s : Sizevalues()) {

double d = sgetPricingFactor();

Systemoutprintln(s + Size has pricing factor of + d);

}

}

}

如果回头想想前面描述过的 getDeclaringClass() 方法您就能明白为什么这些特定于常量的主体和 getClass() 能够在拥有特定于常量的类主体的同时返回不同的类

EnumMap 和 EnumSet

javautil 程序包中包含两个类EnumMap 和 EnumSet这两个类有助于使处理枚举类型变得更容易一些EnumMap 类提供了 javautilMap 接口的一个特殊实现该接口中的键(key)是一个枚举类型EnumSet 类提供了 javautilSet 接口的实现该接口保存了某种枚举类型的值的集合

清单 展示了 EnumMap 类的用法在创建映射时必须为枚举的键传入这个类

清单 EnumMap 的例子

import javautil*;

public class EnumMapSample {

enum Size {

Small

Medium

Large;

}

public static void main(String args[]) {

Map<Size Double> map = new EnumMap<Size Double>(Sizeclass);

mapput(SizeSmall );

mapput(SizeMedium );

mapput(SizeLarge );

for (MapEntry<Size Double> entry : mapentrySet()) {

helper(entry);

}

}

private static void helper(MapEntry<Size Double> entry) {

Systemoutprintln(Map entry: + entry);

}

}

枚举集合的作用类似于特性的集合或者类似于某个枚举类型的所有元素的值的子集EnumSet 类拥有以下一系列的静态方法可以用这些方法从枚举类型中获取单个元素

public static <E extends Enum<E>> EnumSet<E> allOf(Class<E> elementType)

public static <E extends Enum<E>> EnumSet<E> complementOf(EnumSet<E> s)

public static <E extends Enum<E>> EnumSet<E> copyOf(Collection<E> c)

public static <E extends Enum<E>> EnumSet<E> noneOf(Class<E> elementType)

public static <E extends Enum<E>> EnumSet<E> of(E e)

public static <E extends Enum<E>> EnumSet<E> of(E first E rest)

public static <E extends Enum<E>> EnumSet<E> of(E e E e)

public static <E extends Enum<E>> EnumSet<E> of(E e E e E e)

public static <E extends Enum<E>> EnumSet<E> of(E e E e E e E e)

public static <E extends Enum<E>> EnumSet<E> of(E e E e E e E e E e)

public static <E extends Enum<E>> EnumSet<E> range(E from E to)

一旦创建了 EnumSet就可以像对待其他任何 Set 对象那样对待这组对象

结束语

使用枚举

上一篇:为何会堵塞

下一篇:UBB编辑器原来就这么简单