Java 枚举(enum) 详解7种常见的用法 原理 枚举占用内存的原因

用法:周一到周五

给 enum 自定义属性和方法

给 enum 对象加一下 value 的属性和 getValue() 的方法:

package com.hmw.test;

 

/**

 * 枚举测试类

 *

 * @author <a href="mailto:hemingwang0902@126.com">何明旺</a>

 */

public enum EnumTest {

    MON(1), TUE(2), WED(3), THU(4), FRI(5), SAT(6) {

        @Override

        public boolean isRest() {

            return true;

        }

    },

    SUN(0) {

        @Override

        public boolean isRest() {

            return true;

        }

    };

 

    private int value;

 

    private EnumTest(int value) {

        this.value = value;

    }

 

    public int getValue() {

        return value;

    }

 

    public boolean isRest() {

        return false;

    }

}

public class Test {

    public static void main(String[] args) {

        System.out.println("EnumTest.FRI 的 value = " + EnumTest.FRI.getValue());

    }

}

 

原理分析

        enum 的语法结构尽管和 class 的语法不一样,但是经过编译器编译之后产生的是一个class文件。该class文件经过反编译可以看到实际上是生成了一个类,该类继承了java.lang.Enum<E>。EnumTest 经过反编译(javap com.hmw.test.EnumTest 命令)之后得到的内容如下:

public class com.hmw.test.EnumTest extends java.lang.Enum{

    public static final com.hmw.test.EnumTest MON;

    public static final com.hmw.test.EnumTest TUE;

    public static final com.hmw.test.EnumTest WED;

    public static final com.hmw.test.EnumTest THU;

    public static final com.hmw.test.EnumTest FRI;

    public static final com.hmw.test.EnumTest SAT;

    public static final com.hmw.test.EnumTest SUN;

    static {};

    public int getValue();

    public boolean isRest();

    public static com.hmw.test.EnumTest[] values();

    public static com.hmw.test.EnumTest valueOf(java.lang.String);

    com.hmw.test.EnumTest(java.lang.String, intint, com.hmw.test.EnumTest);

}

所以,实际上 enum 就是一个 class,只不过 java 编译器帮我们做了语法的解析和编译而已。

来源: https://www.cnblogs.com/hyl8218/p/5088287.html

 

总结

    可以把 enum 看成是一个普通的 class,它们都可以定义一些属性和方法,不同之处是:enum 不能使用 extends 关键字继承其他类,因为 enum 已经继承了 java.lang.Enum(java是单一继承)。

 

枚举的内存分析?和缺点?

  枚举一般是用来列举一系列相同类型的常量,它是一种特殊的数据类型,使用枚举能够确保参数的安全性。但是Android开发文档上指出,使用枚举会比使用静态变量多消耗两倍的内存

原因:继承了一个类Enum:

二、分析

 

定义一个枚举如下:

 

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

 

可以看到,枚举占用内存的大小比静态变量多得多。

 

三、枚举的替代方案(2种方式

 

 

  • 方法一:使用接口变量

接口变量默认都是public static final的,个人理解接口只是对一类事物的属性和行为更高层次的抽象。对修改关闭,对扩展(不同的实现implements)开放,接口是对开闭原则的一种体现。

 

 

来源: https://www.cnblogs.com/eryaketang/p/8807742.html

 

  •  

 

 

一般情况下,我们可以使用注解来替代枚举,用法如下:

 

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,否则编译器就会报错。

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值