为什么要用枚举

在博客系统中,一篇文章有且可能有这几种状态, 数据库中article文章表中state字段存储数值,表示其状态:

  • 0(已发表Published)
  • 1(草稿Draft)
  • 2(撤回撤回(Delete)

文章实体类中用整数类型的state实例变量标识状态:

  1. public class Article {
  2. /* 文章状态可能值:0,1,2 */
  3. private int state;
  4. ...
  5. }

Service层调用DAO层修改文章状态为‘已发表’:

  1. /**
  2. * dao接口修改文章状态方法
  3. * @param articleId 文章ID
  4. * @param state 状态
  5. */
  6. int updateState(int articleId, int state);
  7. // `Service`层修改文章状态的调用Dao代码:
  8. articleDao.updateState(id, 0);

以上代码有两个问题:

  1. state参数传递并没有限定范围(0,1,2);
  2. 传递数据参数的代码,缺少语义,不看文档或注释不知道0是什么含义;

先来解决第二个问题, 在JDK1.5前常用的解决方式:

  1. /**
  2. * 定义了文章的状态
  3. */
  4. public interface ArticleState{
  5. // 发布状态
  6. int PUBLISHED = 0;
  7. // 草稿状态
  8. int DRAFT = 1;
  9. // 撤回状态
  10. int DELETE = 2;
  11. }

此时修改文章状态的代码:

  1. articleDao.updateState(id, ArticleState.PUBLISHED);

然而此处没有限制必须通过ArticleState传递参数,JDK1.5后提供了枚举来解决这类问题。

Java中声明

在java中,使用enum关键字声明枚举类

  1. /**
  2. * 文章状态枚举类
  3. */
  4. public enum ArticleStateEnum{
  5. PUBLISHED,
  6. DRAFT,
  7. DELETE;
  8. }

然后修改DAO接口:

  1. /**
  2. * dao接口修改文章状态方法
  3. * @param articleId 文章ID
  4. * @param state 状态
  5. */
  6. int updateState(int id, ArticleStateEnum state);

接着Service调用:

  1. // 修改文章状态为发表
  2. articleDao.updateState(id, ArticleStateEnum.PUBLISHED);

以上代码语义清晰,现在传递参数的类型为ArticleStateEnum, 解决了之前描述的两个问题

枚举的本质

使用JDK附带工具javap反编译生枚举类字节码, 注javap反编译只会得到public成员:

看反编译的得到的代码:

  1. class声明,意味着枚举的本质也是类;
  2. 父类声明为java.lang.Enum<>, 意味着枚举类不允许显式使用extends声明父类,包括声明为java.lang.Enum<>也会报错;
  3. 枚举常量,通过public static final修饰符实现,ArticlestateEnum类型声明,意味着所有枚举常量本质是当前枚举类的对象;
  4. values()方法valueOf(String)方法;

这些转换工作是javac编译器帮我们实现的,JVM并不知道枚举的存在,javac帮我们做了一些语法上的转化、简化程序员编程,这种方式称为语法糖。

枚举类VS普通类

枚举类就是类,按照这个逻辑来测试下它和普通类的差别

添加构造函数:

红色行提示编译错误“找不到这样的构造函数”,常量声明处添加参数,如下代码正确:

  1. public enum ArticleStateEnum{
  2. PUBLISHED(0, "已发布"),
  3. DRAFT(1, "草稿"),
  4. DELETE(2, "撤销");
  5. /** 代表的数值 */
  6. private int value;
  7. /** 信息提示 */
  8. private String message;
  9. ArticleStateEnum(int value, String message) {
  10. this.value = value;
  11. this.message = message;
  12. }
  13. // get方法
  14. }

可以推测到常量声明的地方,等价于调用构造函数,通常我们都会为字段添加GET方法不添加SET方法,保证枚举常量的不变性。

枚举类VS普通类的不同点:

  1. 不可以显示声明继承关系;
  2. 常量声明,等价调用构造方法;
  3. 允许有多个构造方法,但修饰符有且仅是private;
  4. 其他地方同类一般无二,可以添加自定添加(方法、字段,抽象成员), 实现接口;

枚举类VS匿名类

看看以下如此夸张的写法,也能编译成功:

观察生成的字节码文件:

把枚举常量声明的地方替换成构造方法调用new ArticleStateEnum(v1, m1),这不就是匿名类的声明吗!

现在向枚举类内添加抽象方放,看看结果:

编译报错“提示有抽象方法未实现”,验证了前面的猜想,这是匿名类的实现,不过不可以显式的使用使用匿名实现枚举类的方式!

常用方法

详细参见API文档Enum类:

  1. public enum ArticleStateEnum{
  2. PUBLISHED,
  3. DRAFT,
  4. DELETE;
  5. public static void main(String[] args) {
  6. ArticleStateEnum[] states = ArticleStateEnum.values(); // 1. 获得所有枚举常量
  7. for(ArticleStateEnum state: states) {
  8. System.out.println("序号:" + state.ordinal() + " 名字:" + state); //2. 输出声明序号和名称
  9. }
  10. System.out.println("......................................");
  11. ArticleStateEnum draft = ArticleStateEnum.valueOf("DRAFT"); //3. 获得某个枚举常量,依据字符串
  12. if(ArticleStateEnum.DRAFT == draft) {
  13. System.out.println(ArticleStateEnum.valueOf("DRAFT").name()); //4. name方法输出名字
  14. }
  15. }
  16. }

输出...

  1. 序号:0 名字:PUBLISHED
  2. 序号:1 名字:DRAFT
  3. 序号:0 名字:DELETE
  4. DRAFT

JAVA中枚举的缺点

java中枚举给我们带来强大的语义的时候,又由于枚举常量对象的本质,给我们带了来庞大性,不如C语言的枚举的轻量:

  1. 枚举常量不可以像C语言一样使用移位运算。
  2. 枚举常量和外部交互麻烦,比如:
      1. mybatis中保存带有枚举字段的实体时,需要你编写转化器(除非按照默认的声明顺序);
      1. 转化为JSON数据时;
    • Spring MVC对请求参数封装时,需要自定义转换器;
  3. 枚举常量无法继承,意味着相似的枚举类之间无法继承,导致产生冗余代码;

建议无特殊情况还是使用枚举常量,毕竟软件的正确性是最重要的

Java 枚举详解的更多相关文章

  1. java枚举详解

    枚举的本质是类,枚举是用来构建常量数据结构的模板(初学者可以以此方式理解: public static final X=xxx),枚举的使用增强了程序的健壮性,在引用一个不存在的枚举值的时候,编译器会 ...

  2. Java Annotation详解 理解和使用Annotation

    系统中用到了java注解: 查了一下如何使用注解,到底注解是什么: (1)创建方法:MsgTrace Java Class==> 在Create New Class中: name:输入MsgTr ...

  3. Java IO 详解

    Java IO 详解 初学java,一直搞不懂java里面的io关系,在网上找了很多大多都是给个结构图草草描述也看的不是很懂.而且没有结合到java7 的最新技术,所以自己来整理一下,有错的话请指正, ...

  4. java关键字(详解)

    目录 1. 基本类型 1) boolean 布尔型 2) byte 字节型 3) char 字符型 4) double 双精度 5) float 浮点 6) int 整型 7) long 长整型 8) ...

  5. Java集合详解3:Iterator,fail-fast机制与比较器

    Java集合详解3:Iterator,fail-fast机制与比较器 今天我们来探索一下LIterator,fail-fast机制与比较器的源码. 具体代码在我的GitHub中可以找到 https:/ ...

  6. Java集合详解3:一文读懂Iterator,fail-fast机制与比较器

    <Java集合详解系列>是我在完成夯实Java基础篇的系列博客后准备开始写的新系列. 这些文章将整理到我在GitHub上的<Java面试指南>仓库,更多精彩内容请到我的仓库里查 ...

  7. Java集合详解1:一文读懂ArrayList,Vector与Stack使用方法和实现原理

    本文非常详尽地介绍了Java中的三个集合类 ArrayList,Vector与Stack <Java集合详解系列>是我在完成夯实Java基础篇的系列博客后准备开始写的新系列. 这些文章将整 ...

  8. 9种Java单例模式详解(推荐)

    单例模式的特点 一个类只允许产生一个实例化对象. 单例类构造方法私有化,不允许外部创建对象. 单例类向外提供静态方法,调用方法返回内部创建的实例化对象.  懒汉式(线程不安全) 其主要表现在单例类在外 ...

  9. Java内部类详解

    Java内部类详解 说起内部类这个词,想必很多人都不陌生,但是又会觉得不熟悉.原因是平时编写代码时可能用到的场景不多,用得最多的是在有事件监听的情况下,并且即使用到也很少去总结内部类的用法.今天我们就 ...

随机推荐

  1. 来一波CSS兼容问题小总结吧

    1.DOCTYPE 影响 CSS 处理; 2.火狐 谷歌等浏览器 设置 padding 后, div 会增加 height 和 width, 但 IE 不会, 故需要用 !important 多设一个 ...

  2. java类初始化,使用构造方法

    public class test { /** * java类的初步学习: *   学会使用和类名相同的两种构造方法,对公共类方法的调用: */ public static void main(Str ...

  3. 读书笔记 effective c++ Item 13 用对象来管理资源

    1.不要手动释放从函数返回的堆资源 假设你正在处理一个模拟Investment的程序库,不同的Investmetn类型从Investment基类继承而来, class Investment { ... ...

  4. Java设计和实现方法

    方法签名 方法名是驼峰命名 方法名最好能说明该方法主要做什么 方法参数的名字最好能说明该参数的意义 方法参数个数最好低于6个 例如: public void setTitleVisible(int l ...

  5. javascript的方法

    1. decodeURIComponent() decodeURIComponent() 函数可对 encodeURIComponent() 函数编码的 URI 进行解码. 语法: decodeURI ...

  6. C++用new创建对象和不用new创建对象的区别解析

    在C++用new创建对象和不用new创建对象是有区别的,不知你是否清楚的了解它们到底有什么样的区别呢?下面小编就用示例来告诉大家吧,需要的朋友可以过来参考下   我们都知道C++中有三种创建对象的方法 ...

  7. 5_jQuery选择器

    <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN""http://www.w3.org/TR/html4/stric ...

  8. BZOJ 1337: 最小圆覆盖1336: [Balkan2002]Alien最小圆覆盖(随机增量法)

    今天才知道有一种东西叫随机增量法就来学了= = 挺神奇的= = A.令ci为包括前i个点的最小圆,若第i+1个点无法被ci覆盖,则第i+1个点一定在ci+1上 B.令ci为包括前i个点的最小圆且p在边 ...

  9. Java Web(七) JSTL标签库

    在之前我们学过在JSP页面上为了不使用脚本,所以我们有了JSP内置的行为.行为只能提供一小部分的功能,大多数的时候还是会用java脚本,接着就使用了EL表达式,基本上EL表达式看似能满足我们的要求,它 ...

  10. linux下apache,php的安装

    apache的安装 1.下载httpd-2.4.16.tar.gz, apr-1.5.2.tar.gz,apr-util-1.5.4.tar.gz,pcre-8.37.zip,解压 2.注意看apac ...