当前位置: 首页 > news >正文

什么是java枚举?为什么要用java枚举?

什么是java枚举?

原始的接口定义常量

public interface IConstants {String MON = "Mon";String TUE = "Tue";String WED = "Wed";String THU = "Thu";String FRI = "Fri";String SAT = "Sat";String SUN = "Sun";
}

语法(定义)

创建枚举类型要使用 enum 关键字,隐含了所创建的类型都是 java.lang.Enum 类的子类(java.lang.Enum 是一个抽象类)。枚举类型符合通用模式 Class Enum<E extends Enum>,而 E 表示枚举类型的名称。枚举类型的每一个值都将映射到 protected Enum(String name, int ordinal) 构造函数中,在这里,每个值的名称都被转换成一个字符串,并且序数设置表示了此设置被创建的顺序。

/*** 枚举测试类* @author <a href="mailto:hemingwang0902@126.com">何明旺</a>*/
public enum EnumTest {MON, TUE, WED, THU, FRI, SAT, SUN;
}

这段代码实际上调用了7次 Enum(String name, int ordinal):

new Enum<EnumTest>("MON",0);
new Enum<EnumTest>("TUE",1);
new Enum<EnumTest>("WED",2);... ...

遍历、switch 等常用操作

对enum进行遍历和switch的操作示例代码:


public class Test {public static void main(String[] args) {for (EnumTest e : EnumTest.values()) {System.out.println(e.toString());}System.out.println("----------------我是分隔线------------------");EnumTest test = EnumTest.TUE;switch (test) {case MON:System.out.println("今天是星期一");break;case TUE:System.out.println("今天是星期二");break;// ... ...default:System.out.println(test);break;}}
}
输出结果:
MON
TUE
WED
THU
FRI
SAT
SUN

enum 对象的常用方法介绍

  • int compareTo(E o)
    • 比较此枚举与指定对象的顺序。
  • Class getDeclaringClass()
    • 返回与此枚举常量的枚举类型相对应的 Class 对象。
  • String name()
    • 返回此枚举常量的名称,在其枚举声明中对其进行声明。
  • int ordinal()
    • 返回枚举常量的序数(它在枚举声明中的位置,其中初始常量序数为零)。
  • String toString()
    • 返回枚举常量的名称,它包含在声明中。
  • static <T extends Enum> T valueOf(Class enumType, String name)
    • 返回带指定名称的指定枚举类型的枚举常量。

public class Test {public static void main(String[] args) {EnumTest test = EnumTest.TUE;//compareTo(E o)switch (test.compareTo(EnumTest.MON)) {case -1:System.out.println("TUE 在 MON 之前");break;case 1:System.out.println("TUE 在 MON 之后");break;default:System.out.println("TUE 与 MON 在同一位置");break;}//getDeclaringClass()System.out.println("getDeclaringClass(): " + test.getDeclaringClass().getName());//name() 和  toString()System.out.println("name(): " + test.name());System.out.println("toString(): " + test.toString());//ordinal(), 返回值是从 0 开始System.out.println("ordinal(): " + test.ordinal());}
}
输出结果:
TUE 在 MON 之后
getDeclaringClass(): com.hmw.test.EnumTest
name(): TUE
toString(): TUE
ordinal(): 1

给 enum 自定义属性和方法

给 enum 对象加一下 value 的属性和 getValue() 的方法:

public enum EnumTest {MON(1), TUE(2), WED(3), THU(4), FRI(5), SAT(6) {@Overridepublic boolean isRest() {return true;}},SUN(0) {@Overridepublic boolean isRest() {return true;}};private int value;private EnumTest(int value) {this.value = value;}public int getValue() {return value;}public boolean isRest() {return false;}
}
public class Test {public static void main(String[] args) {System.out.println("EnumTest.FRI 的 value = " + EnumTest.FRI.getValue());}
}
输出结果:
EnumTest.FRI 的 value = 5

EnumSet,EnumMap 的应用

public class Test {public static void main(String[] args) {// EnumSet的使用EnumSet<EnumTest> weekSet = EnumSet.allOf(EnumTest.class);for (EnumTest day : weekSet) {System.out.println(day);}// EnumMap的使用EnumMap<EnumTest, String> weekMap = new EnumMap(EnumTest.class);weekMap.put(EnumTest.MON, "星期一");weekMap.put(EnumTest.TUE, "星期二");Iterator<Entry<EnumTest, String>> iter = weekMap.entrySet().iterator()// ... ...while (iter.hasNext()) {Entry<EnumTest, String> entry = iter.next();System.out.println(entry.getKey().name() + ":" + entry.getValue());}}
}

原理分析

enum 的语法结构尽管和 class 的语法不一样,但是经过编译器编译之后产生的是一个class文件。该class文件经过反编译可以看到实际上是生成了一个类,该类继承了java.lang.Enum。EnumTest 经过反编译(javap com.hmw.test.EnumTest 命令)之后得到的内容如下:

public class com.hmw.test.EnumTest extends java.lang.Enum{public static final com.hmw.test.EnumTest MON;public static final com.hmw.test.EnumTest TUE;public static final com.hmw.test.EnumTest WED;public static final com.hmw.test.EnumTest THU;public static final com.hmw.test.EnumTest FRI;public static final com.hmw.test.EnumTest SAT;public static final com.hmw.test.EnumTest SUN;static {};public int getValue();public boolean isRest();public static com.hmw.test.EnumTest[] values();public static com.hmw.test.EnumTest valueOf(java.lang.String);com.hmw.test.EnumTest(java.lang.String, int, int, com.hmw.test.EnumTest);
}

所以,实际上 enum 就是一个 class,只不过 java 编译器帮我们做了语法的解析和编译而已。

values()方法,在编译过程中产生

总结

可以把 enum 看成是一个普通的 class,它们都可以定义一些属性和方法,不同之处是:enum 不能使用 extends 关键字继承其他类,因为 enum 已经继承了 java.lang.Enum(java是单一继承)。

为什么要用java枚举?

需求背景

假设现在有两种订单类型:预订订单和非预订订单。

  • 需求一: 方法submitOrder根据不同订单类型进行不同的处理。
  • 需求二: 给一个对象: OrderResult设置订单类型。

实现方式

1.不使用枚举

public class EnumTest {private static final int ORDER_TYPE1 = 1;private static final int ORDER_TYPE2 = 2;public void submitOrder(int orderType,OrderResult orderResult){orderResult.setType(orderType);if(orderType == ORDER_TYPE1){System.out.println("订单类型1处理方法");}else if(orderType == ORDER_TYPE2){System.out.println("订单类型2处理方法");}}public static void main(String[] args) {EnumTest test = new EnumTest();test.submitOrder(3, new OrderResult());}
}

从代码中可以看到几点缺点:

  1. int参数不做类型检查,可以给上面的submitOrder方法传入任意int值
  2. 代码可读性低,如果没有文档或者源码,不知道给submitOrder传递的值的意义。

很多人在使用int枚举时,并没有像上例中将int值定义成static final。如果忘记定义变量为final则int枚举的值就可以被修改,如果忘记定义变量为static,就可能出现使用这个int值时,该int值还没有被初始化好。

2.枚举实现

public class EnumTest1 {private enum ORDER_TYPE{ORDER_TYPE1(1),ORDER_TYPE2(2);private final int value;private ORDER_TYPE(int value){this.value = value;}}public void submitOrder(ORDER_TYPE orderType,OrderResult orderResult){orderResult.setType(orderType.value);if(orderType == ORDER_TYPE.ORDER_TYPE1){System.out.println("订单类型1处理方法");}else if(orderType == ORDER_TYPE.ORDER_TYPE2){System.out.println("订单类型2处理方法");}}public static void main(String[] args) {EnumTest1 test = new EnumTest1();test.submitOrder(ORDER_TYPE.ORDER_TYPE1, new OrderResult());}
}
  1. 编译器将对enum进行类型检查,类型不符合的编译器会直接报错。
  2. 相比与int枚举型直接传int数值的方式,enum传递enum类型对象的方式,代码可读性更高,传递的枚举类型一目了然。
  3. enum类型中的对象本身就是static final的。

误区

private enum ORDER_TYPE{ORDER_TYPE1(1),ORDER_TYPE2(2);private final int value;private ORDER_TYPE(int value){this.value = value;}}

enum本质也是一个类,所以方法ORDER_TYPE(int value)是这个枚举类型的构造函数,故每个枚举类型在初始化的时候需要给构造函数传递响应的值,如: ORDER_TYPE1(1)。

这种情况下,想得到枚举类型对应的int数值时可以通过ORDER_TYPE.ORDER_TYPE1.value获取。

有些人可能会直接使用enum中的ordinal()方法直接实现enum类型与int类型的关联。ordinal()方法返回的是enum类型在被定义时的序数,如:

ORDER_TYPE.ORDER_TYPE1.value.ordinal()返回值为0。所以获取枚举类型对应的int数值貌似也可以通过ORDER_TYPE.PRE_ORDER.value.ordinal()+1实现。

序数是很不可靠的东西,序数是可以改变的,假设有一天ORDER_TYPE的定义变了,需要增加几种类型,或者不小心换了ORDER_TYPE1和ORDER_TYPE2定义时的顺序,这时就会造成很严重的bug,而且不好发现,编译时,运行时都不会有报错

http://www.lryc.cn/news/234618.html

相关文章:

  • USB复合设备构建CDC+HID鼠标键盘套装
  • 准备篇(四)HTTP 基本原理
  • 模板初阶笔记分享
  • 使用Spring Boot实现大文件断点续传及文件校验
  • 读取PDF中指定数据写入EXCEL文件
  • [黑马程序员SpringBoot2]——开发实用篇1
  • Python------列表 集合 字典 推导式(本文以 集合为主)
  • 网工内推 | Linux运维,六险二金,最高30K,IE认证优先
  • 服务器集群配置LDAP统一认证高可用集群(配置tsl安全链接)-centos9stream-openldap2.6.2
  • 12-1- GAN -简单网络-线性网络
  • Antv/G2 分组柱状图+折线图双轴图表
  • springboot323基于Java的美妆购物网站的设计与实现
  • vue项目本地开发完成后部署到服务器后报404
  • Android设计模式--状态模式
  • C++关系运算符重载
  • HLS基础issue
  • C#特性(Attribute)
  • 【设计模式】七大设计原则
  • 思维导图软件 Xmind mac中文版特点介绍
  • Day32力扣打卡
  • 抗击.Elbie勒索病毒:如何应对.Elbie病毒威胁,保卫您的数据
  • Vue3 函数式弹窗
  • 如何解决 Critical dependency: the request of a dependency is an expression ?
  • 挑战视觉边界,探索图形验证码背后的黑科技
  • 【网络奇缘】- 计算机网络|网络类型|性能指标
  • Leetcode—剑指Offer LCR 140.训练计划II【简单】
  • 梦想编织者——Adobe Dreamweaver
  • springMVC学习笔记-请求映射,参数绑定,响应,restful,响应状态码,springMVC拦截器
  • Python实现视频字幕时间轴格式转换
  • 蓝桥杯 枚举