用法六:使用接口组织枚举
public interface Food { enum Coffee implements Food{ BLACK_COFFEE,DECAF_COFFEE,LATTE,CAPPUCCINO } enum Dessert implements Food{ FRUIT, CAKE, GELATO } }
/** * 测试继承接口的枚举的使用(by 大师兄 or 大湿胸。) */ private static void testImplementsInterface() { for (Food.DessertEnum dessertEnum : Food.DessertEnum.values()) { System.out.print(dessertEnum + " "); } System.out.println(); //我这地方这么写,是因为我在自己测试的时候,把这个coffee单独到一个文件去实现那个food接口,而不是在那个接口的内部。 for (CoffeeEnum coffee : CoffeeEnum.values()) { System.out.print(coffee + " "); } System.out.println(); //搞个实现接口,来组织枚举,简单讲,就是分类吧。如果大量使用枚举的话,这么干,在写代码的时候,就很方便调用啦。 //还有就是个“多态”的功能吧, Food food = Food.DessertEnum.CAKE; System.out.println(food); food = CoffeeEnum.BLACK_COFFEE; System.out.println(food); }
运行结果:
用法七:关于枚举集合的使用
java.util.EnumSet和java.util.EnumMap是两个枚举集合。EnumSet保证集合中的元素不重复;EnumMap中的 key是enum类型,而value则可以是任意类型。关于这个两个集合的使用就不在这里赘述,可以参考JDK文档。
关于枚举的实现细节和原理请参考:
参考资料:《ThinkingInJava》第四版 http://softbeta.iteye.com/blog/1185573
附:案例
package com.lxk.enumTest; /** * Java枚举用法测试 */ public class EnumTest { public static void main(String[] args) { forEnum(); useEnumInJava(); } /** * 循环枚举,输出ordinal属性;若枚举有内部属性,则也输出。(说的就是我定义的TYPE类型的枚举的typeName属性) */ private static void forEnum() { for (SimpleEnum simpleEnum : SimpleEnum.values()) { System.out.println(simpleEnum + " ordinal " + simpleEnum.ordinal()); } System.out.println("------------------"); for (TYPE type : TYPE.values()) { System.out.println("type = " + type + " type.name = " + type.name() + " typeName = " + type.getTypeName() + " ordinal = " + type.ordinal()); } } /** * 在Java代码使用枚举 */ private static void useEnumInJava() { String typeName = "f5"; TYPE type = TYPE.fromTypeName(typeName); if (TYPE.BALANCE.equals(type)) { System.out.println("根据字符串获得的枚举类型实例跟枚举常量一致"); } else { System.out.println("大师兄代码错误"); } } /** * 季节枚举(不带参数的枚举常量)这个是最简单的枚举使用实例 * Ordinal 属性,对应的就是排列顺序,从0开始。 */ private enum SimpleEnum { SPRING, SUMMER, AUTUMN, WINTER } /** * 常用类型(带参数的枚举常量,这个只是在书上不常见,实际使用还是很多的,看懂这个,使用就不是问题啦。) */ private enum TYPE { FIREWALL("firewall"), SECRET("secretMac"), BALANCE("f5"); private String typeName; TYPE(String typeName) { this.typeName = typeName; } /** * 根据类型的名称,返回类型的枚举实例。 * * @param typeName 类型名称 */ public static TYPE fromTypeName(String typeName) { for (TYPE type : TYPE.values()) { if (type.getTypeName().equals(typeName)) { return type; } } return null; } public String getTypeName() { return this.typeName; } } }
运行结果:
简单的例子,大家基本都用过,看不懂的基本都是第二个例子。可以看到,在第二个例子里面,后面带有参数,其实可以这么理解。
enum这个关键字,可以理解为跟class差不多,这也个单独的类。可以看到,上面的例子里面有属性,有构造方法,有getter,也可以有setter,但是一般都是构造传参数。还有其他自定义方法。那么在这些东西前面的,以逗号隔开的,最后以分号结尾的,这部分叫做,这个枚举的实例。也可以理解为,class new 出来的实例对象。这下就好理解了。只是,class,new对象,可以自己随便new,想几个就几个,而这个enum关键字,他就不行,他的实例对象,只能在这个enum里面体现。也就是说,他对应的实例是有限的。这也就是枚举的好处了,限制了某些东西的范围,举个栗子:一年四季,只能有春夏秋冬,你要是字符串表示的话,那就海了去了,但是,要用枚举类型的话,你在enum的大括号里面把所有的选项,全列出来,那么这个季节的属性,对应的值,只能在里面挑。不能有其他的。
我上面的例子,就是根据typeName,你可以从那些例子里面挑选到唯一的一个TYPE类型的枚举实例--TYPE.BALANCE。注意方法
TYPE type = TYPE.fromTypeName(typeName);
这个方法的返回类型就是这个TYPE枚举类型的。
这下就好理解,这个枚举是怎么在工作了吧
再补充一下:
上面那个带参数的枚举类型的实例里面实际上是三个属性,除了我自定义的typeName以外,还有2个是系统自带的。看下面源码的图:
看到这里之后,不知道你能不能理解下面图片内说明的话:下面图片主要说明在使用枚举时,的规范和标准。希望可以在实际开发时候用到。
最后补充一点:
也许你知道呢,但是也许你不知道呢?我是真的不知道,测了之后才知道!!!
枚举类型对象之间的值比较,是可以使用==,直接来比较值,是否相等的,不是必须使用equals方法的哟。
有的老铁,说这个switch case怎么写,我就在下面再啰嗦一下。
private static void testSwitchCase() { String typeName = "f5"; //这几行注释呢,你可以试着三选一,测试一下效果。 //String typeName = "firewall"; //String typeName = "secretMac"; TypeEnum typeEnum = TypeEnum.fromTypeName(typeName); if (typeEnum == null) { return; } switch (typeEnum) { case FIREWALL: System.out.println("枚举名称(即默认自带的属性 name 的值)是:" + typeEnum.name()); System.out.println("排序值(默认自带的属性 ordinal 的值)是:" + typeEnum.ordinal()); System.out.println("枚举的自定义属性 typeName 的值是:" + typeEnum.getTypeName()); break; case SECRET: System.out.println("枚举名称(即默认自带的属性 name 的值)是:" + typeEnum.name()); System.out.println("排序值(默认自带的属性 ordinal 的值)是:" + typeEnum.ordinal()); System.out.println("枚举的自定义属性 typeName 的值是:" + typeEnum.getTypeName()); break; case BALANCE: System.out.println("枚举名称(即默认自带的属性 name 的值)是:" + typeEnum.name()); System.out.println("排序值(默认自带的属性 ordinal 的值)是:" + typeEnum.ordinal()); System.out.println("枚举的自定义属性 typeName 的值是:" + typeEnum.getTypeName()); break; default: System.out.println("default"); } }
运行结果:
老铁们,看完这个枚举,你要懂个概念,那就是,这个枚举,他是个对象,就像你定义的Student类,Person类,等等一些个类一样。
要有这么个概念。只要是个类,他就可以有构造函数,可以有属性,可以有方法。
对的,老铁,你对这个属性,构造函数啥的,有概念吧,没有的话,我可就郁闷啦。
然后,你就看到,这个地方有2个默认的属性,一个是name,一个是ordinal,这2个属性就像你定义Student类和Person类的name和age一样,
只不过,这2个是系统自带的属性,不用你自己去定义啦。
你也可以给这个枚举类,也就是你自己声明的枚举,随便加属性。
我上面代码例子里面的那个TypeEnum那个枚举,就是这么干的,就简单的添加了个自定义属性typeName,
虽然他有自己的name了,那姑且叫我这个自定义的属性叫别名吧。
可以看到,我例子里面就是通过自己写的那个构造方法给我这个自定义的属性初始化值的。
还有,这个构造方法是不可以,也不被运行public的,不信,你可以试试。
还有,你不能对系统自带的name属性,在构造函数里面赋值,没有为什么。