用enum代替int常量

1. int枚举:

引入枚举前,一般是声明一组具名的int常量,每个常量代表一个类型成员,这种方法叫做int枚举模式。

int枚举模式是类型不安全的,例如下面两组常量:性别和动物种类,二者不存在任何关系,然而却可以将ANIMAL_DOG传入一个需要性别参数的方法中,编译器不会出现警告,而且方法内部逻辑还会继续执行比较等操作。

采用int枚举模式的程序是十分脆弱的,因为int枚举是编译时常量,被编译到使用它们的客户端中,如果与枚举常量关联的int值发生变化客户端就必须重新编译,然而不重新编译却不会报错,但是会导致程序的结果不准确,例如上面的常量SEX_MAN被客户端使用,于是将其值1编译到客户端的.class中,然后如果API类中将SEX_MAN的值改为2,却不重新编译客户端,那么客户端得到的结果就是不准确的了。

int枚举常量很难被翻译成可打印的字符串,不利于开发调试。

开发过程中还可能遇到这种模式的变体,如String枚举模式,同样是存在上述问题。

2. 枚举类型:

由一组固定的常量组成的合法值的类型,例如:

实现思路:通过公有静态的final域为每个枚举常量导出实例的类。

枚举类型是类型安全的枚举模式,而且完善类int枚举模式的不足。

枚举类型还允许添加任意的方法和域,并实现任意的接口,它提供类所以Object方法的高级实现,实现了Comparable和Serializable接口,并针对枚举类型的可任意改变性设计了序列化方式。

特定于常量的方法实现:

在枚举类型中声明一个抽象方法,并在特定于常量的类主体中,用具体的方法覆盖每个常量的抽象方法。例如下面这样定义一个代表加减乘除等运算符的枚举。

策略枚举:多个枚举常量同时共享相同的行为时,考虑使用策略枚举。

例如下面这样:

用实例域代替序数

序数:枚举天生就与一个单独的int值相关联,所有枚举都有一个ordinal()方法,返回每个枚举常量在类型中的数字位置(类似于数组索引)。

永远不要根据枚举的序数导出与他相关联的值,而是将它保存在一个实例域中 (Enum规范中关于ordinal()写到:"大多数程序员都不需要这个方法,它是设计成用于像EnumSet,EnumMap这种基于枚举的通用数据结构的)。

用EnumSet代替位域

位域:可以用or(|)位运算将几个常量合并到一个集合中,例如下面代码这样:

位域的不足:具有int枚举的所有缺点

替代方案--EnumSet:从单个枚举类型中提取多个值,每个EnumSet内容都表示为位矢量, 如果底层的枚举类型有64或更少的元素(大多如此),整个EnumSet就是用单个long来表示, 因此,它的性能比得上位域的性能

用EnumMap代替序数索引

使用序数索引ordinal的场景:

上面代码实现了对花园中对植物进行分类,然而存在许多问题:

1.数组不能与泛型兼容,需要进行未受检对转换;

2.set数组并不知道每个索引set的set代表什么;

3. 之前有提到不推荐使用ordinal;

解决方案:EnumMap

用接口模拟可伸缩的枚举:

虽然无法编写可扩展的枚举类型,却可以通过编写接口以及实现该接口的基础枚举类型,对它进行模拟,这样允许客户端编写自己的枚举来实现接口;如果API是根据接口编写的,那么在使用基础枚举类型的任何地方,也都可以使用这些枚举。例如下面代码,还是用之前的算数运算符举例:(但是这样还是有些不足,就是无法实现从一个枚举类型继承到另一个枚举类型,代码少的当然可以直接复制粘贴, 如果功能比较多则可以将他们封装在一个辅助类或静态辅助方法中,避免代码的复制工作)。

注解优先于命名模式

- 命名模式:有些程序元素需要通过某种工具或框架进行特殊处理

例1:JUnit测试框架原本要求用户一定要用test作为测试方法名的开头

例2:iOS中的init方法要求必须是initXXX()

- 命名模式缺陷:

1. 文字拼写错误会导致失败,且没有任何提示,造成错误的安全感,如JUnit的测试方法testXX写成textXX或tsetXX等

2. 无法确保他们只用于相应的程序元素,如JUnit的命名只对方法生效,将某个类命名testXX是无效的,不会报错,但不会执行测试

3. 没有提供将参数值与程序元素关联起来的好方法,如JUnit想增加一种测试类别,只在抛出某种特定异常时才会成功, 而这个异常类型需要用户通过参数进行自定义,这种实现通过命名模式实现(将异常类型编写到方法名中)并不理想。

注解对上面问题的解决,请看下面代码:

坚持使用Override注解

应该在想要覆盖超类声明的每个方法声明中使用Override注解 例如我们经常会重写自定义模型类的equals方法,下面用代码说明使用Override注解的优势。

使用Override还有一点好处,就是可以区分哪些方法是超类对,哪些方法子类扩展对

用标记接口定义类型

标记接口:没有方法声明,只是表示具有某种属性,如Serializable接口

标记接口的优点 :

1. 标记接口定义的类型是由被标记类的实例实现的,标记注解则没有这样的类型

2. 标记接口可以更加精确的被锁定,可以是对其他接口的扩展,也可以被其他标记接口扩展,如Collection和Set

标记注解的优点:1. 可以通过默认方式添加一个或多个注解类型的元素,给已被使用的注解类型添加更多的信息,方便扩展

2. 另一个优点在于它们是更大的注解机制的一部分,因此,标记注解,在那些支持注解作为编程元素之一的框架中同样具有一致性

如何选择?

- 如果标记是应用到任何程序元素而不只是类或接口,那就必须使用注解

- 如果只是用于类或接口,需要考虑要编写只接受有这种标记的方法,使用接口作为相关方法的参数类型, 可以提供编译时就进行类型检查的好处

- 是否要永远限制这个标记只用于特殊接口的元素,如果是,最好将标记定义成该接口的一个子接口

我是今阳,如果想要进阶和了解更多的干货,欢迎关注公众号”今阳说“接收我的最新文章

Java注解参数类型枚举值_EffectiveJava-5-枚举和注解相关推荐

  1. JAVA语言规范 JAVA SE 8 - 类型、值和变量

    JAVA语言规范 JAVA SE 8 - 类型.值和变量 类型和值的种类 简单类型和值 整数类型和值 整数操作 浮点数类型.格式和值 浮点数操作 boolean类型和布尔值 引用类型和值 对象 Obj ...

  2. java.lang.date_无法将java.lang.String类型的值转换为必需类型java.util.Date

    from date: to date: Trouver 这是控制器代码部分: @RequestParam(name = "d1", defaultValue = "190 ...

  3. Java的byte类型取值范围为什么是负128到正127呢

    Java的byte类型取值范围是-128~127为什么负数是128正数到127呢 概念:java中用补码表示二进制数,补码的最高位是符号位,最高位为"0"表示正数,最高位为&quo ...

  4. Java可变参数类型实例

    可变参数:         Java1.5增加了新特性:可变参数:适用于参数个数不确定,类型确定的情况,java把可变参数当做数组处理. 可变的参数类型,也称为不定参数类型.英文缩写是varargus ...

  5. C# 声明枚举、枚举值描述 获取枚举值字符串 根据枚举获取枚举值描述

    一.声明枚举值.枚举值描述 using System.ComponentModel;public enum status {/// <summary>/// 取消/// </summ ...

  6. java接口参数类型为枚举_Spring MVC处理参数中的枚举类型通用实现方法

    前言 在开发的过程中,会涉及到大量的枚举类型数据,我们都知道,Springmvc本身能自动转换很多的数据类型,也支持你自定义转换类型,非常灵活. 本文主要介绍的是关于Spring MVC处理参数的枚举 ...

  7. java值传递和引用传递_辨析Java方法参数中的值传递和引用传递

    小方法大门道 小瓜瓜作为一个Java初学者,今天跟我说她想通过一个Java方法,将外部变量通过参数传递到方法中去,进行逻辑处理,方法执行完毕之后,再对修改过的变量进行判断处理,代码如下所示. publ ...

  8. 深度解析Java可变参数类型以及与数组的区别

    这篇文章主要介绍了Java方法的可变参数类型,通过实例对Java中的可变参数类型进行了较为深入的分析,需要的朋友可以参考下. Java方法中的可变参数类型是一个非常重要的概念,有着非常广泛的应用.本文 ...

  9. java 限制参数类型_java定义受限制的类型参数操作

    有时您可能想限制可以在参数化类型中用作类型参数的类型. 例如,对数字进行操作的方法可能只希望接受Number或其子类的实例. 这就是有界类型参数的用途. 受限制参数类型的方法示例 要声明有界类型参数, ...

最新文章

  1. 独家 | 图解BiDAF中的单词嵌入、字符嵌入和上下文嵌入(附链接)
  2. XDebug分析php代码性能
  3. 给书配代码-电力经济调度(3):计及网络安全约束的经济调度模型
  4. apache php mysql 整合_PHP+Apache+MySQL整合
  5. 扫地机器人隔板_【扫地机器人使用】_摘要频道_什么值得买
  6. 简单c语言课设计题目,C语言课程设计题目
  7. 二、mongodb数据库系列——聚合操作 索引操作 权限管理
  8. 【每日SQL打卡】​​​​​​​​​​​​​​​DAY 8丨判断三角形【难度简单】
  9. java 反射获取修饰符_java之反射和BeanUtils类
  10. SQL Server删除语句
  11. python 在window 系统 连接并操作远程 oracle 数据库
  12. 调用DB2存储过程出现错误
  13. F2FS文件系统论文解读
  14. 从《色戒》,看人性的欲望
  15. Kali渗透-MSF木马免杀技术
  16. 路由器ipv4和ipv6转发原理
  17. Doom3 引擎渲染管线分析
  18. solidworks属性管理器_发现SOLIDWORKS自定义属性(下)
  19. 几款U盘数据恢复的软件使用测评
  20. informix-系统视图

热门文章

  1. 如何将SQL GROUP BY和聚合转换为Java 8
  2. 在Spring中使用Future对象调用Async方法调用
  3. JavaFX:太空侵略者在175 LOC以下
  4. Couchbase 2.0归类视图简介
  5. Java:检查器框架
  6. 在多节点集群中运行Cassandra
  7. macOS安装Maven_IDEA集成Maven
  8. MacOS下如何通过命令搜索文件和打开文件
  9. Windows 如何在命令终端(CMD)使用命令来访问本地/远程的 Oracle 数据库呢?
  10. openfire消息通知推送_APP消息推送功能之前端后台设计