一、概述
枚举一般是用来列举一系列相同类型的常量,它是一种特殊的数据类型,使用枚举能够确保参数的安全性。但是Android开发文档上指出,使用枚举会比使用静态变量多消耗两倍的内存,应该尽量避免在Android中使用枚举,那么枚举为什么会更消耗内存呢?下面一起分析一下。
二、分析
定义一个枚举如下:
package com.liunian.androidbasic.enumtest;
/**
* Created by dell on 2018/4/19.
* 测试枚举占用内存
*/
public enum Sex {
MAN, WOMAN;
}
1、使用javac将其编译成.class文件,命令为:javac Sex.java;
2、用jad反编译.class文件,生成Sex.jad,命令为jad Sex.class,jad的下载地址:http://www.javadecompilers.com/jad
打开Sex.jad文件:
package com.liunian.androidbasic.enumtest;
public final class Sex extends Enum
{
public static Sex[] values()
{
return (Sex[])$VALUES.clone();
}
public static Sex valueOf(String s)
{
return (Sex)Enum.valueOf(com/liunian/androidbasic/enumtest/Sex, s);
}
private Sex(String s, int i)
{
super(s, i);
}
public static final Sex MAN;
public static final Sex WOMAN;
private static final Sex $VALUES[];
static
{
MAN = new Sex("MAN", 0);
WOMAN = new Sex("WOMAN", 1);
$VALUES = (new Sex[] {
MAN, WOMAN
});
}
}
从反编译的代码来看,我们定义的枚举,编译器会将其转换成一个类,这个类继承自java.lang.Enum类,除此之外,编译器还会帮我们生成多个枚举类的实例,赋值给我们定义的枚举类型常量,并且还声明了一个枚举对象的数组,保存了所有的枚举对象。下面我们分别来计算一下采用静态变量和枚举占用内存的大小。
a、静态变量
public final static int MAN = 0;
public final static int WOMAN = 1;
int占用内存大小为4,加起来占用内存大小为8
b、枚举
首先我们看一下枚举对象占用的内存大小
public abstract class Enum<E extends Enum<E>> implements Comparable<E>, Serializable {
private final String name;
private final int ordinal;
}
作为 Enum 成员变量 name(对象引用) 和 ordinal(int) 它们各占用 4 个字节,该对象实例占用:
12 + 4 + 4 = 20bytes,对齐之后是 24 字节
其中12个字节是对象头占用的内存,Enum中包含了一个String类型的对象,空字符串对象本身就是32 字节,加上其中的字符数组最少也会占据 24 个字节, 对字符串加字符数组最少会占据 56 个字节,故一个 Enum 实例,最少 56+24 = 80个字节。在加上字符串"MAN",“WOMAN”占用的空间,两个枚举对象占用的空间大小为:
(4 + 80 + 4)* 2 + 3 + 5 = 184
枚举类里面还声明了数组Sex $VALUES[],占用大小为24 + 4 + 4 = 32
故,枚举总共占用内存大小为184 + 32 = 216
可以看到,枚举占用内存的大小比静态变量多得多。
三、枚举的替代方案
一般情况下,我们可以使用注解来替代枚举,用法如下:
1、在build.gradle依赖注解库
compile 'com.android.support:support-annotations:24.2.0'
2、定义内容
package com.liunian.androidbasic.enumtest;
import android.support.annotation.IntDef;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* Created by dell on 2018/4/19.
*/
@IntDef({Sex.MAN, Sex.WOMAN})
@Retention(RetentionPolicy.SOURCE)
public @interface Sex {
public static final int MAN = 0;
public static final int WOMAN = 1;
}
3、使用
public void setSex(@SexOne int sex) {
}
setSex(SexOne.WOMAN);
在调用setSex时,只能传入SexOne.MAN和SexOne.WOMAN,否则编译器就会报错。