欢迎访问 生活随笔!

尊龙凯时首页

当前位置: 尊龙凯时首页 > 编程语言 > java >内容正文

java

java枚举(深刻而不深沉平淡而不平庸) -尊龙凯时首页

发布时间:2024/10/14 java 25 豆豆
尊龙凯时首页 收集整理的这篇文章主要介绍了 java枚举(深刻而不深沉平淡而不平庸) 小编觉得挺不错的,现在分享给大家,帮大家做个参考.

1.枚举概述
2.枚举的常用方法
3.枚举的特性
4.枚举实现接口和类
5.switch状态机,组织枚举,策略枚举
6.enumset和enummap

1.1枚举知识导图

1.2枚举格式及使用说明

注:如果枚举中没有定义方法,也可以在最后一个实例后面加逗号、分号或什么都不加。如果要为enum定义方法,那么必须在enum的最后一个实例尾部添加一个分号

下面三种声明方式是等价的:

enum color { red, green, blue } enum color { red, green, blue, } enum color { red, green, blue; }

看一个枚举格式的一个实例

enum color {red, green, blue }

如果枚举不添加任何方法,枚举值默认为从0开始的有序数值。以 color 枚举类型举例,它的枚举常量依次为 red:0,green:1,blue:2

1.3枚举的好处和应用场景

枚举的好处:可以将常量组织起来,统一进行管理
枚举的典型应用场景:错误码、状态机等

1.枚举的本质:

尽管 enum 看起来像是一种新的数据类型,事实上,enum是一种受限制的类,并且具有自己的方法。
创建enum时,编译器会为你生成一个相关的类,这个类继承自 java.lang.enum,但是枚举不可以继承enum 不可以继承另外一个类,当然,也不能继承另一个 enum 。因为 enum 实际上都继承自 java.lang.enum 类,而 java 不支持多重继承,所以 enum 不能再继承其他类,当然也不能继承另一个 enum。

看以下代码:

public enum colorenum{red,white,blue,black,green }

经过编译器编译后产生的是一个class文件,该class文件经过反编译软件编译后可以看到实际上生成了一个类:

public class com.com.yarward.design.colorenum extends java.lang.enum{public static final com.com.yarward.design.colorenum red;public static final com.com.yarward.design.colorenum white;public static final com.com.yarward.design.colorenum blue;public static final com.com.yarward.design.colorenum black;public static final com.com.yarward.design.colorenum green;static {};public static com.com.yarward.design.colorenum[] values();public static com.com.yarward.design.colorenum valueof(java.lang.string);.... }

我们再来看一下enum类的源码:

public abstract class enum<e extends enum<e>>implements comparable<e>, serializable { ... }

2.1常用方法法介绍

方法名作用
values()返回 enum 实例的数组,而且该数组中的元素严格保持在 enum 中声明时的顺序。
name()返回实例名。
ordinal()返回实例声明时的次序,从0开始。
getdeclaringclass()返回实例所属的 enum 类型。
equals()判断是否为同一个对象。

此外,java.lang.enum实现了comparable和 serializable 接口,所以也提供 compareto() 方法。

2.2方法的是使用实例:

package untl; public class enummethoddemo {enum color {red, green, blue;}enum size {big, middle, small;}public static void main(string args[]) {system.out.println("=========== print all color ===========");for (color c : color.values()) {system.out.println(c " ordinal: " c.ordinal());}system.out.println("=========== print all size ===========");for (size s : size.values()) {system.out.println(s " ordinal: " s.ordinal());}color green = color.green;system.out.println("green name(): " green.name());system.out.println("green getdeclaringclass(): " green.getdeclaringclass());system.out.println("green hashcode(): " green.hashcode());system.out.println("green compareto color.green: " green.compareto(color.green));system.out.println("green equals color.green: " green.equals(color.green));system.out.println("green equals size.middle: " green.equals(size.middle));system.out.println("green equals 1: " green.equals(1));system.out.format("green == color.blue: %b\n", green == color.blue);} } 运行结果: =========== print all color =========== red ordinal: 0 green ordinal: 1 blue ordinal: 2 =========== print all size =========== big ordinal: 0 middle ordinal: 1 small ordinal: 2 green name(): green green getdeclaringclass(): class untl.enummethoddemo$color green hashcode(): 1163157884 green compareto color.green: 0 green equals color.green: true green equals size.middle: false green equals 1: false green == color.blue: false

除了不能继承,基本上可以将 enum 看做一个常规的类。

3.1关于枚举的成员变量和成员方法

在c/c 语言中的enum,可以用赋值符号=显示的为枚举常量赋值;但是 ,很遗憾,java 语法中却不允许使用赋值符号 = 为枚举常量赋值。

例:c/c 语言中的枚举声明:

typedef enum{one = 1,two,three = 3,ten = 10 } number;

枚举可以添加普通方法、静态方法、抽象方法、构造方法
java 虽然不能直接为实例赋值,但是它有更优秀的尊龙凯时首页的解决方案:为 enum 添加方法来间接实现显示赋值。唯一用起来比较别扭的一点是,enum相当于把类的实例化(枚举常量)放到了类声明的内部,这些枚举常量也可以重写枚举类型中的方法。

创建 enum 时,可以为其添加多种方法,甚至可以为其添加构造方法。

注意一个细节:如果要为enum定义方法,那么必须在enum的最后一个实例尾部添加一个分号。此外,在enum中,必须先定义实例,不能将字段或方法定义在实例前面。否则,编译器会报错。

例子:

public enum errorcodeen {ok(0, "成功"),error_a(100, "错误a"),error_b(200, "错误b");//相当于创建了3个实例,调用了3次 enum(string name, int ordinal)errorcodeen(int number, string description) {this.code = number;this.description = description;}private int code;private string description;public int getcode() {return code;}public string getdescription() {return description;}public static void main(string args[]) { // 静态方法for (errorcodeen s : errorcodeen.values()) {system.out.println("code: " s.getcode() ", description: " s.getdescription());}} } 运行结果: code: 0, description: 成功 code: 100, description: 错误a code: 200, description: 错误b

4.1实现接口

public interface inumberenum {int getcode();string getdescription(); }public enum errorcodeen2 implements inumberenum {ok(0, "成功"),error_a(100, "错误a"),error_b(200, "错误b");errorcodeen2(int number, string description) {this.code = number;this.description = description;}private int code;private string description;@overridepublic int getcode() {return code;}@overridepublic string getdescription() {return description;} }

4.2实现类

本例和上例效果相同。

public class plant2 {public enum vegetable implements inumberenum {...} // 省略代码public enum fruit implements inumberenum {...} // 省略代码 }

5.1.switch状态机:

jdk7以后,switch已经支持int、char、string、enum类型的参数。这几种类型的参数比较起来,使用枚举的switch代码更具有可读性。

enum signal {red, yellow, green} public static string gettrafficinstruct(signal signal) {string instruct = "信号灯故障";switch (signal) {case red:instruct = "红灯停";break;case yellow:instruct = "黄灯请注意";break;case green:instruct = "绿灯行";break;default:break;}return instruct; }

5.2.组织枚举:

可以将类型相近的枚举通过接口或类组织起来。但是一般用接口方式进行组织。原因是:java接口在编译时会自动为enum类型加上public static修饰符;java类在编译时会自动为 enum 类型加上static修饰符。看出差异了吗?没错,就是说,在类中组织 enum,如果你不给它修饰为 public,那么只能在本包中进行访问。

例:在接口中组织 enum

public interface plant {enum vegetable implements inumberenum {potato(0, "土豆"),tomato(0, "西红柿");vegetable(int number, string description) {this.code = number;this.description = description;}private int code;private string description;@overridepublic int getcode() {return 0;}@overridepublic string getdescription() {return null;}}enum fruit implements inumberenum {apple(0, "苹果"),orange(0, "桔子"),banana(0, "香蕉");fruit(int number, string description) {this.code = number;this.description = description;}private int code;private string description;@overridepublic int getcode() {return 0;}@overridepublic string getdescription() {return null;}} }

5.3.策略枚举

effectivejava中展示了一种策略枚举。这种枚举通过枚举嵌套枚举的方式,将枚举常量分类处理。
这种做法虽然没有switch语句简洁,但是更加安全、灵活。

例:effectviejava中的策略枚举范例

enum payrollday {monday(paytype.weekday), tuesday(paytype.weekday), wednesday(paytype.weekday), thursday(paytype.weekday), friday(paytype.weekday), saturday(paytype.weekend), sunday(paytype.weekend);private final paytype paytype;payrollday(paytype paytype) {this.paytype = paytype;}double pay(double hoursworked, double payrate) {return paytype.pay(hoursworked, payrate);}// 策略枚举private enum paytype {weekday {double overtimepay(double hours, double payrate) {return hours <= hours_per_shift ? 0 : (hours - hours_per_shift)* payrate / 2;}},weekend {double overtimepay(double hours, double payrate) {return hours * payrate / 2;}};private static final int hours_per_shift = 8;abstract double overtimepay(double hrs, double payrate);double pay(double hoursworked, double payrate) {double basepay = hoursworked * payrate;return basepay overtimepay(hoursworked, payrate);}} }

测试

system.out.println(“时薪100的人在周五工作8小时的收入:” payrollday.friday.pay(8.0, 100));
system.out.println(“时薪100的人在周六工作8小时的收入:” payrollday.saturday.pay(8.0, 100));

java 中提供了两个方便操作enum的工具类——enumset和 enummap。
enumset 是枚举类型的高性能 set 实现。它要求放入它的枚举常量必须属于同一枚举类型.enummap 是专门为枚举类型量身定做的 map 实现。虽然使用其它的 map 实现(如hashmap)也能完成枚举类型实例到值得映射,但是使用 enummap 会更加高效:它只能接收同一枚举类型的实例作为键值,并且由于枚举类型实例的数量相对固定并且有限,所以 enummap 使用数组来存放与枚举类型对应的值。这使得 enummap 的效率非常高

// enumset的使用system.out.println("enumset展示"); enumset<errorcodeen> errset = enumset.allof(errorcodeen.class); for (errorcodeen e : errset) {system.out.println(e.name() " : " e.ordinal()); }// enummap的使用 system.out.println("enummap展示"); enummap<statemachine.signal, string> errmap = new enummap(statemachine.signal.class); errmap.put(statemachine.signal.red, "红灯"); errmap.put(statemachine.signal.yellow, "黄灯"); errmap.put(statemachine.signal.green, "绿灯"); for (iterator<map.entry<statemachine.signal, string>> iter = errmap.entryset().iterator(); iter.hasnext();) {map.entry<statemachine.signal, string> entry = iter.next();system.out.println(entry.getkey().name() " : " entry.getvalue()); }

总结

以上是尊龙凯时首页为你收集整理的java枚举(深刻而不深沉平淡而不平庸)的全部内容,希望文章能够帮你解决所遇到的问题。

如果觉得尊龙凯时首页网站内容还不错,欢迎将尊龙凯时首页推荐给好友。

网站地图