android开发Enum (枚举)的更轻量级的替代方案 —— @IntDef的使用 原创 2017年11月28日 09:17:58 77 最近在翻看Android源码的时候,无意中看到这么一个注解,

android开发Enum (枚举)的更轻量级的替代方案 —— @IntDef的使用

原链接

在翻看Android源码的时候,无意中看到这么一个注解,@IntDef 好奇心重的我,查看了一下注释,原来是用来替换掉枚举的!他比int更安全,比枚举更轻量!
开始之前我们先看看Android 官方文档中的一段话。
Be careful with code abstractions

Developers often use abstractions simply as a good programming practice, because 
abstractions can improve code flexibility and maintenance. However, abstractions 
come at a significant cost: generally they require a fair amount more code that 
needs to be executed, requiring more time and more RAM for that code to be mapped
into memory. So if your abstractions aren't supplying a significant benefit, you 
should avoid them.

For example, enums often require more than twice as much memory as static constants. You should strictly avoid using enums on Android.

  
  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
官方的说法就是:我们在写代码的时候要注意类型的使用,以便于提高代码的扩展性和维护性,但是原型的使用一般会付出更多的内存的代价,所以如果没有特别大的好处,要尽量避免使用。对于枚举来说占用的内存往往是使用静态常量的两倍,因而我们要尽量避免在Android中使用枚举。
因而使用@IntDef注解来代替枚举是个不错的选择。
首先,添加android注解依赖:
compile 'com.android.support:support-annotations:25.1.0'

  
  
  • 1
  • 2
具体用法如下:
public class MainActivity extends AppCompatActivity{

  public static final int STATE_NONE = -1;
  public static final int STATE_LOADING = 0;
  public static final int STATE_SUCCESS = 1;
  public static final int STATE_ERROR = 2;
  public static final int STATE_EMPTY = 3;

  private @State int state;

  public void setState(@State int state){
      this.state = state;
  } 

  @State
  public int getState() {
    return this.state;
  }

  @IntDef({STATE_EMPTY, STATE_ERROR, STATE_LOADING, STATE_NONE, STATE_SUCCESS})
  @Retention(RetentionPolicy.SOURCE)
  public @interface State {
  }
}

 @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
  }
附录:

使用 Enum 的缺点

每一个枚举值都是一个对象,在使用它时会增加额外的内存消耗,所以枚举相比与 Integer 和 String 会占用更多的内存。

较多的使用 Enum 会增加 DEX 文件的大小,会造成运行时更多的开销,使我们的应用需要更多的空间。

如果你的应用使用很多的 Enum ,最好使用Integer 或 String 替代他们,但是这样还会有问题。

既然都说到这个份上了,那么有什么比较好的解决方法呢?


解决方案

既然是因为参数的类型太泛了造成的类型不安全,那么我只要将参数限定在某一个类型集合里面,不就大功告成了?!

是滴,一下就是要将的@IntDef/@StringDef + @interface来进行限定参数。

使用方法:

  官方文档说明,安卓开发应避免使用Enum(枚举类),因为相比于静态常量Enum会花费两倍以上的内存。参 http://developer.android.com/training/articles/memory.html#Overhead

    那么如果需要使用Enum应该怎么做呢?

    https://noobcoderblog.wordpress.com/2015/04/12/java-enum-and-android-intdefstringdef-annotation/

[java]  view plain  copy
  1. public class MainActivity extends Activity {  
  2.     
  3.     //先定义 常量  
  4.     public static final int SUNDAY = 0;  
  5.     public static final int MONDAY = 1;  
  6.     public static final int TUESDAY = 2;  
  7.     public static final int WEDNESDAY = 3;  
  8.     public static final int THURSDAY = 4;  
  9.     public static final int FRIDAY = 5;  
  10.     public static final int SATURDAY = 6;  
  11.     
  12.     //用 <span></span>@IntDef "包住" 常量;  
  13.     // @Retention 定义策略  
  14.     // 声明构造器  
  15.     @IntDef({SUNDAY, MONDAY,TUESDAY,WEDNESDAY,THURSDAY,FRIDAY,SATURDAY})  
  16.     @Retention(RetentionPolicy.SOURCE)  
  17.     public @interface WeekDays {}  
  18.     
  19.     @WeekDays int currentDay = SUNDAY;  
  20.     
  21.     @Override  
  22.     protected void onCreate(Bundle savedInstanceState) {  
  23.         super.onCreate(savedInstanceState);  
  24.         setContentView(R.layout.activity_main);  
  25.         setCurrentDay(WEDNESDAY);  
  26.     
  27.         //声明变量  
  28.         @WeekDays int today = getCurrentDay();  
  29.     
  30.         switch (today){  
  31.             case SUNDAY:  
  32.                 break;  
  33.             case MONDAY:  
  34.                 break;  
  35.             case TUESDAY:  
  36.                 break;  
  37.             case WEDNESDAY:  
  38.                 break;  
  39.             case THURSDAY:  
  40.                 break;  
  41.             case FRIDAY:  
  42.                 break;  
  43.             case SATURDAY:  
  44.                 break;  
  45.             default:  
  46.                 break;  
  47.         }  
  48.     
  49.     }  
  50.     
  51.     public void setCurrentDay(@WeekDays int currentDay) {  
  52.         this.currentDay = currentDay;  
  53.     }  
  54.     
  55.     @WeekDays  
  56.     public int getCurrentDay() {  
  57.         return currentDay;  
  58.     }  
  59. }  

使用注解库

这些注解不是默认加载的,它们被包装为一个单独的库。Support Library现在是由一些更小的库组成的,包括:v4-support、appcompat、gridlayout、mediarouter等等。
添加注解的最简单的方法就是打开Project Structure对话框。首先在左边选中module,然后右边选中Dependencies标签,点击“+”号按钮,选择Library Dependency。如果SDK中已经包括了Android Support库,那么注解支持库就会显示在快捷选择列表中了,只需要点击选择就可以。
步骤1:点击Project Structure按钮



步骤2:选中Dependencies标签,点击“+”号按钮


步骤3:在下拉列表中选中support-annotations库


点击OK确定,这将会修改build.gradle文件。当然也可以手动在Gradle中添加如下依赖:
[java]  view plain  copy
  1. dependencies {  
  2.     compile 'com.android.support:support-annotations:23.1.0'  
  3. }  
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值