一、概述

        枚举一般是用来列举一系列相同类型的常量,它是一种特殊的数据类型,使用枚举能够确保参数的安全性。但是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,否则编译器就会报错。