`

【转】Java 语言中 Enum 类型的使用介绍

阅读更多

Enum 类型的介绍


枚举类型(Enumerated Type) 很早就出现在编程语言中,它被用来将一组类似的值包含到一种类型当中。而这种枚举类型的名称则会被定义成独一无二的类型描述符,在这一点上和常量的定义相似。不过相比较常量类型,枚举类型可以为申明的变量提供更大的取值范围。
举个例子来说明一下,如果希望为彩虹描绘出七种颜色,你可以在 Java 程序中通过常量定义方式来实现。


清单 1. 常量定义

Java代码 
  1.             
  2. Public static class RainbowColor {
  3.   
  4.   // 红橙黄绿青蓝紫七种颜色的常量定义
  5.   public static final int RED = 0;
  6.   public static final int ORANGE = 1;
  7.   public static final int YELLOW = 2;
  8.   public static final int GREEN = 3;
  9.   public static final int CYAN = 4;
  10.   public static final int BLUE = 5;
  11.   public static final int PURPLE = 6;
  12. }

使用的时候,你可以在程序中直接引用这些常量。但是,这种方式还是存在着一些问题。
1.类型不安全由于颜色常量的对应值是整数形,所以程序执行过程中很有可能给颜色变量传入一个任意的整数值,导致出现错误。2.没有命名空间由于颜色常量只是类的属性,当你使用的时候不得不通过类来访问。
3.一致性差因为整形枚举属于编译期常量,所以编译过程完成后,所有客户端和服务器端引用的地方,会直接将整数值写入。这样,当你修改旧的枚举整数值后或者增加新的枚举值后,所有引用地方代码都需要重新编译,否则运行时刻就会出现错误。
4.类型无指意性由于颜色枚举值仅仅是一些无任何含义的整数值,如果在运行期调试时候,你就会发现日志中有很多魔术数字,但除了程序员本身,其他人很难明白其奥秘。

如何定义 Enum 类型

为了改进 Java 语言在这方面的不足弥补缺陷,5.0 版本 SDK 发布时候,在语言层面上增加了枚举类型。枚举类型的定义也非常的简单,用 enum 关键字加上名称和大括号包含起来的枚举值体即可,例如上面提到的彩虹颜色就可以用新的 enum 方式来重新定义:

Java代码 
  1. enum RainbowColor { RED, ORANGE, YELLOW, GREEN, CYAN, BLUE, PURPLE }

从上面的定义形式来看,似乎 Java 中的枚举类型很简单,但实际上 Java 语言规范赋予枚举类型的功能非常的强大,它不仅是简单地将整形数值转换成对象,而是将枚举类型定义转变成一个完整功能的类定义。这种类型定义的扩展允许开发者给枚举类型增加任何方法和属性,也可以实现任意的接口。另外,Java 平台也为 Enum 类型提供了高质量的实现,比如默认实现 Comparable 和 Serializable 接口,让开发者一般情况下不用关心这些细节。回到本文的主题上来,引入枚举类型到底能够给我们开发带来什么样好处呢?一个最直接的益处就是扩大 switch 语句使用范围。5.0 之前,Java 中 switch 的值只能够是简单类型,比如 int、long、char, 有了枚举类型之后,就可以使用对象了。这样一来,程序的控制选择就变得更加的方便,看下面的例子:

清单 2. 定义 Enum 类型

Java代码 
  1. // 定义一周七天的枚举类型        
  2. public enum WeekDayEnum { Mon, Tue, Wed, Thu, Fri, Sat, Sun }
  3. // 读取当天的信息
  4. WeekDayEnum today = readToday();
  5. // 根据日期来选择进行活动
  6. switch(today) {
  7.   Mon: do something; break;
  8.   Tue: do something; break;
  9.   Wed: do something; break;
  10.   Thu: do something; break;
  11.   Fri: do something; break;
  12.   Sat: play sports game; break;
  13.   Sun: have a rest; break;
  14. }

对于这些枚举的日期,JVM 都会在运行期构造成出一个简单的对象实例一一对应。这些对象都有唯一的 identity,类似整形数值一样,switch 语句就根据此来进行执行跳转。

如何定制 Enum 类型


除了以上这种最常见的枚举定义形式外,如果需要给枚举类型增加一些复杂功能,也可以通过类似 class 的定义来给枚举进行定制。比如要给 enum 类型增加属性,可以像下面这样定义:

清单 3. 定制枚举类型

Java代码 
  1. // 定义 RSS(Really Simple Syndication) 种子的枚举类型
  2. public enum NewsRSSFeedEnum {
  3.   // 雅虎头条新闻 RSS 种子
  4.   YAHOO_TOP_STORIES("http://rss.news.yahoo.com/rss/topstories"),
  5.   
  6.   //CBS 头条新闻 RSS 种子
  7.   CBS_TOP_STORIES("http://feeds.cbsnews.com/CBSNewsMain?format=xml"),
  8.   
  9.   // 洛杉矶时报头条新闻 RSS 种子
  10.   LATIMES_TOP_STORIES("http://feeds.latimes.com/latimes/news?format=xml");
  11.     
  12.   // 枚举对象的 RSS 地址的属性
  13.   private String rss_url;
  14.     
  15.   // 枚举对象构造函数
  16.   private NewsRSSFeedEnum(String rss) {
  17.     this.rss_url = rss;
  18.   }
  19.     
  20.   // 枚举对象获取 RSS 地址的方法
  21.   public String getRssURL() {
  22.     return this.rss_url;
  23.   }
  24. }

上面头条新闻的枚举类型增加了一个 RSS 地址的属性 , 记录头条新闻的访问地址。同时,需要外部传入 RSS 访问地址的值,因而需要定义一个构造函数来初始化此属性。另外,还需要向外提供方法来读取 RSS 地址。

 

如何避免错误使用 Enum

不过在使用 Enum 时候有几个地方需要注意:
1.enum 类型不支持 public 和 protected 修饰符的构造方法,因此构造函数一定要是 private 或 friendly 的。也正因为如此,所以枚举对象是无法在程序中通过直接调用其构造方法来初始化的。
2.定义 enum 类型时候,如果是简单类型,那么最后一个枚举值后不用跟任何一个符号;但如果有定制方法,那么最后一个枚举值与后面代码要用分号';'隔开,不能用逗号或空格。
3.由于 enum 类型的值实际上是通过运行期构造出对象来表示的,所以在 cluster 环境下,每个虚拟机都会构造出一个同义的枚举对象。因而在做比较操作时候就需要注意,如果直接通过使用等号 ( ‘ == ’ ) 操作符,这些看似一样的枚举值一定不相等,因为这不是同一个对象实例。
看下面的这个例子:


清单 4. 避免错误使用 Enum 示例

Java代码 
  1. // 定义一个一周七天的枚举类型
  2. package example.enumeration.codes;
  3. public enum WeekDayEnum {
  4. Mon(1), Tue(2), Wed(3), Thu(4), Fri(5), Sat(6), Sun(7);
  5. private int index;
  6. WeekDayEnum(int idx) {
  7. this.index = idx;
  8. }
  9. public int getIndex() {
  10. return index;
  11. }
  12. }
  13. // 客户端程序,将一个枚举值通过网络传递给服务器端
  14. package example.enumeration.codes;
  15. import java.io.IOException;
  16. import java.io.ObjectOutputStream;
  17. import java.io.OutputStream;
  18. import java.net.InetSocketAddress;
  19. import java.net.Socket;
  20. import java.net.UnknownHostException;
  21. public class EnumerationClient {
  22. public static void main(String... args) throws UnknownHostException, IOException {
  23. Socket socket = new Socket();
  24. // 建立到服务器端的连接
  25. socket.connect(new InetSocketAddress("127.0.0.1"8999));
  26. // 从连接中得到输出流
  27. OutputStream os = socket.getOutputStream();
  28. ObjectOutputStream oos = new ObjectOutputStream(os);
  29. // 将星期五这个枚举值传递给服务器端
  30. oos.writeObject(WeekDayEnum.Fri);
  31. oos.close();
  32. os.close();
  33. socket.close();
  34. }
  35. }
  36. // 服务器端程序,将从客户端收到的枚举值应用到逻辑处理中
  37. package example.enumeration.codes;
  38. import java.io.*;
  39. import java.net.ServerSocket;
  40. import java.net.Socket;
  41. public class EnumerationServer {
  42. public static void main(String... args) throws IOException, ClassNotFoundException {
  43. ServerSocket server = new ServerSocket(8999);
  44. // 建立服务器端的网络连接侦听
  45. Socket socket = server.accept();
  46. // 从连接中获取输入流
  47. InputStream is = socket.getInputStream();
  48. ObjectInputStream ois = new ObjectInputStream(is);
  49. // 读出客户端传递来的枚举值
  50. WeekDayEnum day = (WeekDayEnum) ois.readObject();
  51. // 用值比较方式来对比枚举对象
  52. if (day == WeekDayEnum.Fri) {
  53. System.out.println("client Friday enum value is same as server's");
  54. else if (day.equals(WeekDayEnum.Fri)) {
  55. System.out.println("client Friday enum value is equal to server's");
  56. else {
  57. System.out.println("client Friday enum value is not same as server's");
  58. }
  59. // 用 switch 方式来比较枚举对象
  60. switch (day) {
  61. case Mon:
  62. System.out.println("Do Monday work");
  63. break;
  64. case Tue:
  65. System.out.println("Do Tuesday work");
  66. break;
  67. case Wed:
  68. System.out.println("Do Wednesday work");
  69. break;
  70. case Thu:
  71. System.out.println("Do Thursday work");
  72. break;
  73. case Fri:
  74. System.out.println("Do Friday work");
  75. break;
  76. case Sat:
  77. System.out.println("Do Saturday work");
  78. break;
  79. case Sun:
  80. System.out.println("Do Sunday work");
  81. break;
  82. default:
  83. System.out.println("I don't know which is day");
  84. break;
  85. }
  86. ois.close();
  87. is.close();
  88. socket.close();
  89. }
  90. }

打印结果如下:

Java代码 
  1. client Friday enum value is same as server's
  2. Do Friday work

通过程序执行结果,我们能够发现在分布式条件下客户端和服务端的虚拟机上都生成了一个枚举对象,即使看起来一样的 Fri 枚举值,如果使用等号‘ == ’进行比较的话会出现不等的情况。而 switch 语句则是通过 equal 方法来比较枚举对象的值,因此当你的枚举对象较复杂时候,你就需要小心 override 与比较相关的方法,防止出现值比较方面的错误。

 

Enum 相关工具类

JDK5.0 中在增加 Enum 类的同时,也增加了两个工具类 EnumSet 和 EnumMap,这两个类都放在 java.util 包中。EnumSet 是一个针对枚举类型的高性能的 Set 接口实现。EnumSet 中装入的所有枚举对象都必须是同一种类型,在其内部,是通过 bit-vector 来实现,也就是通过一个 long 型数。EnumSet 支持在枚举类型的所有值的某个范围中进行迭代。回到上面日期枚举的例子上:


Java代码 
  1. enum WeekDayEnum { Mon, Tue, Wed, Thu, Fri, Sat, Sun }

你能够在每周七天日期中进行迭代,EnumSet 类提供一个静态方法 range 让迭代很容易完成:

Java代码 
  1. for(WeekDayEnum day : EnumSet.range(WeekDayEnum.Mon, WeekDayEnum.Fri)) {
  2.   System.out.println(day);
  3. }

打印结果如下:


Java代码 
  1. Mon
  2. Tue
  3. Wed
  4. Thu
  5. Fri

EnumSet 还提供了很多个类型安全的获取子集的 of 方法,使你很容易取得子集:


Java代码 
  1. EnumSet<WeekDayEnum> subset = EnumSet.of(WeekDayEnum.Mon, WeekDayEnum.Wed);
  2.   for (WeekDayEnum day : subset) {
  3.     System.out.println(day);  
  4.   }

打印结果如下:

Java代码 
  1. Mon
  2. Wed

与 EnumSet 类似,EnumMap 也是一个高性能的 Map 接口实现,用来管理使用枚举类型作为 keys 的映射表,内部是通过数组方式来实现。EnumMap 将丰富的和安全的 Map 接口与数组快速访问结合到一起,如果你希望要将一个枚举类型映射到一个值,你应该使用 EnumMap。看下面的例子:
清单 5. EnumMap 示例

Java代码 
  1. // 定义一个 EnumMap 对象,映射表主键是日期枚举类型,值是颜色枚举类型
  2. private static Map<WeekDayEnum, RainbowColor> schema =
  3.     new EnumMap<WeekDayEnum, RainbowColor>(WeekDayEnum.class);
  4.   
  5. static{
  6.   // 将一周的每一天与彩虹的某一种色彩映射起来
  7.   for (int i = 0; i < WeekDayEnum.values().length; i++) {
  8.     schema.put(WeekDayEnum.values(), RainbowColor.values());
  9.   }
  10. }
  11. System.out.println("What is the lucky color today?");
  12. System.out.println("It's " + schema.get(WeekDayEnum.Sat));

当你询问周六的幸运色彩时候,会得到蓝色:清单 6. 运行结果

Java代码 
  1. What is the lucky color today?
  2. It's BLUE

结束语

Enum 类型提出给 JAVA 编程带了了极大的便利,让程序的控制更加的容易,也不容易出现错误。所以在遇到需要控制程序流程时候,可以多想想是否可以利用 enum 来实现。


关于作者

刘进目前在 IBM STG 云计算团队从事云管理平台产品开发,拥有十年的 Java 应用开发经验,对 J2SE 和 J2EE 的相关应用开发比较熟悉。

文章来源:developerworks

分享到:
评论

相关推荐

    基于百度地图实现的定位功能.zip

    android 源码学习. 资料部分来源于合法的互联网渠道收集和整理,供大家学习参考与交流。本人不对所涉及的版权问题或内容负法律责任。如有侵权,请通知本人删除。感谢CSDN官方提供大家交流的平台

    加载本地图片,绝对不会出现OOM.zip

    android 源码学习. 资料部分来源于合法的互联网渠道收集和整理,供大家学习参考与交流。本人不对所涉及的版权问题或内容负法律责任。如有侵权,请通知本人删除。感谢CSDN官方提供大家交流的平台

    2015年中国移动电子竞技游戏发展趋势报告(1).zip

    2015年中国移动电子竞技游戏发展趋势报告(1).zip

    CKplayer-v6.8.zip

    ckplayer是一款在网页上播放视频的免费的播放器,功能强大,体积小巧,跨平台,使用起来随心所欲。 CKplayer播放器主要以adobe的flash(所使用的版本是CS5)平台开发,所以在支持flash插件的平台和浏览器上都可以使用,而无需下载其它插件,如果你需要修改完整版里提供的相关的flash源文件,请使用adobe的flash cs5以上版本打开源文件修改。 ckplayer同时也支持

    46.书籍学习平台的设计与实现-Springboot+ Mysql+Java+ B/S结构(可运行源码+数据库+设计文档)论坛

    46.书籍学习平台的设计与实现|Springboot+ Mysql+Java+ B/S结构(可运行源码+数据库+设计文档)论坛,公告,付费专区,免费专区,销售,会员办理,书籍分类 详细设计文档链接:http://t.csdnimg.cn/GSeDN 内容概要: 全套项目源码+详尽文档,一站式解决您的学习与项目需求。 适用人群: 计算机、通信、人工智能、自动化等专业的学生、老师及从业者。 使用场景及目标: 无论是毕设、期末大作业还是课程设计,一键下载,轻松部署,助您轻松完成项目。 项目代码经过调试测试,确保直接运行,节省您的时间和精力。 其他说明: 项目整体具有较高的学习借鉴价值,基础能力强的可以在此基础上修改调整,以实现不同的功能。

    密码学实验报告2.docx

    密码学实验报告2.docx

    各种旋转动画的ImageView(1).zip

    android 源码学习. 资料部分来源于合法的互联网渠道收集和整理,供大家学习参考与交流。本人不对所涉及的版权问题或内容负法律责任。如有侵权,请通知本人删除。感谢CSDN官方提供大家交流的平台

    S7200基本编程指令.ppt

    S7200基本编程指令.ppt

    基于python+OpenCV的火车票识别源码+使用文档+全部资料(优秀项目).zip

    【资源说明】 基于python+OpenCV的火车票识别源码+使用文档+全部资料(优秀项目).zip基于python+OpenCV的火车票识别源码+使用文档+全部资料(优秀项目).zip 【备注】 1、该项目是个人高分毕业设计项目源码,已获导师指导认可通过,答辩评审分达到95分 2、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 3、本项目适合计算机相关专业(如软件工程、计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载使用,也可作为毕业设计、课程设计、作业、项目初期立项演示等,当然也适合小白学习进阶。 4、如果基础还行,可以在此代码基础上进行修改,以实现其他功能,也可直接用于毕设、课设、作业等。 欢迎下载,沟通交流,互相学习,共同进步!

    WordPress.zip

    android 源码学习. 资料部分来源于合法的互联网渠道收集和整理,供大家学习参考与交流。本人不对所涉及的版权问题或内容负法律责任。如有侵权,请通知本人删除。感谢CSDN官方提供大家交流的平台

    移动机器人机械臂的设计小论文.doc

    移动机器人机械臂的设计小论文.doc

    基于Python+OpenCV+tinker的指纹识别系统,使用的硬件为AS608源码+使用文档+全部资料(优秀项目).zip

    【资源说明】 基于Python+OpenCV+tinker的指纹识别系统,使用的硬件为AS608源码+使用文档+全部资料(优秀项目).zip基于Python+OpenCV+tinker的指纹识别系统,使用的硬件为AS608源码+使用文档+全部资料(优秀项目).zip 【备注】 1、该项目是个人高分毕业设计项目源码,已获导师指导认可通过,答辩评审分达到95分 2、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 3、本项目适合计算机相关专业(如软件工程、计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载使用,也可作为毕业设计、课程设计、作业、项目初期立项演示等,当然也适合小白学习进阶。 4、如果基础还行,可以在此代码基础上进行修改,以实现其他功能,也可直接用于毕设、课设、作业等。 欢迎下载,沟通交流,互相学习,共同进步!

    xiuno模板知乎蓝魔改版源码附多个插件.zip

    xiuno模板知乎蓝魔改版源码附多个插件

    2022年 【24页】从孪生到融生,AIGC成为长期方向.zip

    2022年 【24页】从孪生到融生,AIGC成为长期方向.zip

    [信息与通信]使用EMIF将Xilinx_FPGA与TI_DSP平台接口.pdf

    [信息与通信]使用EMIF将Xilinx_FPGA与TI_DSP平台接口.pdf

    基于Opencv+Filterpy实现YOLOV3-SORT车辆跟踪与车流统计算法源码+使用文档+全部资料(优秀项目).zip

    【资源说明】 基于Opencv+Filterpy实现YOLOV3-SORT车辆跟踪与车流统计算法源码+使用文档+全部资料(优秀项目).zip基于Opencv+Filterpy实现YOLOV3-SORT车辆跟踪与车流统计算法源码+使用文档+全部资料(优秀项目).zip基于Opencv+Filterpy实现YOLOV3-SORT车辆跟踪与车流统计算法源码+使用文档+全部资料(优秀项目).zip 【备注】 1、该项目是个人高分毕业设计项目源码,已获导师指导认可通过,答辩评审分达到95分 2、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 3、本项目适合计算机相关专业(如软件工程、计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载使用,也可作为毕业设计、课程设计、作业、项目初期立项演示等,当然也适合小白学习进阶。 4、如果基础还行,可以在此代码基础上进行修改,以实现其他功能,也可直接用于毕设、课设、作业等。 欢迎下载,沟通交流,互相学习,共同进步!

    仿QQ消息列表(ListView)滑动删除效果源码.zip

    android 源码学习. 资料部分来源于合法的互联网渠道收集和整理,供大家学习参考与交流。本人不对所涉及的版权问题或内容负法律责任。如有侵权,请通知本人删除。感谢CSDN官方提供大家交流的平台

    最新招标流程图.pdf

    1.招标投标活动不受地区或者部门的限制。任何单位和个人不得违法限制或者排斥本地区、 本系统以外的法人或者其他组织参加投标,不得以任何方式非法干涉招标投标活动。 2.招标人设有标底,标底必须保密 3.可以不招标的情况之一都可以不招标:①不可替代②采购人自行建设、生产或提供③已通 过招标方式特许的④需要向原中标人采购的,否则影响配套要求的,但是不得超过合同金额 的10%⑤国家规定的其他特殊情况

    C++基于自实现OpenCV图像处理函数的静态车道线检测项目源码+使用文档+全部资料(优秀项目).zip

    【资源说明】 C++基于自实现OpenCV图像处理函数的静态车道线检测项目源码+使用文档+全部资料(优秀项目).zipC++基于自实现OpenCV图像处理函数的静态车道线检测项目源码+使用文档+全部资料(优秀项目).zipC++基于自实现OpenCV图像处理函数的静态车道线检测项目源码+使用文档+全部资料(优秀项目).zip 【备注】 1、该项目是个人高分毕业设计项目源码,已获导师指导认可通过,答辩评审分达到95分 2、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 3、本项目适合计算机相关专业(如软件工程、计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载使用,也可作为毕业设计、课程设计、作业、项目初期立项演示等,当然也适合小白学习进阶。 4、如果基础还行,可以在此代码基础上进行修改,以实现其他功能,也可直接用于毕设、课设、作业等。 欢迎下载,沟通交流,互相学习,共同进步!

    基于单片机的遥控机械臂设计.doc

    基于单片机的遥控机械臂设计.doc

Global site tag (gtag.js) - Google Analytics