前言
看到目前项目里用到蛮多枚举, 才有了这篇小文章分享
为什么使用Enum
java中的Enum是包含固定常量集的数据类型.当我们需要预定义一组代表某种数据的值时一般都会使用枚举, 而当要保证类型安全时, 我们经常会使用Enum。
比如, 当我们要保证常量使用正常时, 我们经常使用Enum在编译时校验确保类型安全
使用Enum的缺点
在Android开发者官网上, 有这样一段话
enums often require more than twice as much memory as static constants. You should strictly avoid using enums on Android.
Enum中的每个值都是一个对象,每个声明都将使用一些运行时内存来简单引用该对象,所以Enum相较于static int会占用更多的内存
.
另外添加单个Enum将增加最终DEX文件的大小
(是static int的13倍).
解决方案
Google提供了注解库通过Typedef
协助我们解决了Enum的问题,它可以确保特定参数, 返回值或字段引用特定的常量集,还可以完成代码以自动提供允许的常量.
我们可以通过使用@IntDef
和@StringDef
来帮助我们在编译时检查像Enum这项的变量赋值.
使用姿势
首先我们需要依赖注解库
1
dependencies {compile 'com.android.support:support-annotations:24.2.0'}
直接上代码了, 因为还是蛮简单的
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25public class Person {
public static final int MALE = 0;
public static final int FEMALE = 1;
private int sex;
public String getSexValue(){
if(MALE == sex){
return "男";
}else if(FEMALE == sex){
return "女";
}
return "";
}
public void setSex(@sexDef int sex) {
this.sex = sex;
}
// 定义该注解被保留的时间长短
// RetentionPolicy.CLASS 注解被保留到class文件, 但jvm加载class文件时候被遗弃, 这是默认生命周期; 用于在编译时进行一些预处理操作, 比如生成一些辅助代码(ButterKnife)
// RetentionPolicy.RUNTIME 注解不仅被保存到class文件中, jvm加载class文件之后, 仍然存在;用于在运行时去动态获取注解信息
// RetentionPolicy.SOURCE 注解只保留在源文件, 当Java文件编译成class文件的时候, 注解被遗弃; 用于做一些检查性操作
(RetentionPolicy.SOURCE)
// 使用@IntDef定义声明常量作为枚举
({MALE, FEMALE})
// 使用@interface声明新的枚举注解类型
public sexDef{}
}当我们调用setSex设置性别的时候, 如果输入非指定类型, 则编译不会通过
总结
与普通static常量相比, Enum的使用至少为整个apk添加至少两倍以上的字节数,并且使用5到10倍的RAM内存。所以建议尽量避免使用Enum, 当需要使用上述特性时,建议以@IntDef 或 @StringDef 替代使用。
更多的可以看看[The price of ENUMs][https://www.youtube.com/watch?v=Hzs6OBcvNQE&feature=youtu.be]视频, 需科学上网