Java注解参数类型枚举值_EffectiveJava-5-枚举和注解

用enum代替int常量

1. int枚举:

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

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

d12ae0f94022

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

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

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

2. 枚举类型:

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

d12ae0f94022

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

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

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

d12ae0f94022

特定于常量的方法实现:

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

d12ae0f94022

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

例如下面这样:

d12ae0f94022

用实例域代替序数

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

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

d12ae0f94022

用EnumSet代替位域

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

d12ae0f94022

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

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

d12ae0f94022

用EnumMap代替序数索引

使用序数索引ordinal的场景:

d12ae0f94022

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

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

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

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

解决方案:EnumMap

d12ae0f94022

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

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

d12ae0f94022

d12ae0f94022

注解优先于命名模式

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

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

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

- 命名模式缺陷:

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

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

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

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

d12ae0f94022

坚持使用Override注解

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

d12ae0f94022

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

用标记接口定义类型

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

标记接口的优点 :

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

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

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

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

如何选择?

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

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

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

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

d12ae0f94022

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值