Java 枚举 enum

JDK1.5引入了新的类型——枚举。在 Java 中它虽然算个“小”功能,却给我的开发带来了“大”方便。

Java 枚举是一个特殊的类,一般表示一组常量,比如一年的 4 个季节,一个年的 12 个月份,一个星期的 7 天,方向有东南西北等。

Java 枚举类使用 enum 关键字来定义,各个常量使用逗号 , 来分割。

例如定义一个颜色的枚举类。

enum Color 
{ 
    RED, GREEN, BLUE; 
} 

以上枚举类 Color 颜色常量有 RED, GREEN, BLUE,分别表示红色,绿色,蓝色。

使用实例:

实例

enum Color
{
    RED, GREEN, BLUE;
}
 
public class Test
{
    // 执行输出结果
    public static void main(String[] args)
    {
        Color c1 = Color.RED;
        System.out.println(c1);
    }
}

执行以上代码输出结果为:

RED
values(), ordinal() 和 valueOf() 方法
enum 定义的枚举类默认继承了 java.lang.Enum 类,并实现了 java.lang.Serializable 和 java.lang.Comparable 两个接口。

values(), ordinal() 和 valueOf() 方法位于 java.lang.Enum 类中:

values() 返回枚举类中所有的值。
ordinal()方法可以找到每个枚举常量的索引,就像数组索引一样。
valueOf()方法返回指定字符串值的枚举常量。
实例

enum Color
{
    RED, GREEN, BLUE;
}
 
public class Test
{
    public static void main(String[] args)
    {
        // 调用 values()
        Color[] arr = Color.values();
 
        // 迭代枚举
        for (Color col : arr)
        {
            // 查看索引
            System.out.println(col + " at index " + col.ordinal());
        }
 
        // 使用 valueOf() 返回枚举常量,不存在的会报错 IllegalArgumentException
        System.out.println(Color.valueOf("RED"));
        // System.out.println(Color.valueOf("WHITE"));
    }
}

执行以上代码输出结果为:

RED at index 0
GREEN at index 1
BLUE at index 2
RED
1.运用

@Getter
public enum LoginCode {

    /**
     * IDM登录码
     */
    LOGIN_SUCCESS(160111,"登录成功!"),
    LOGIN_FAIL(2104111,"账号或密码错误,请重新输入"),
    PERMISSION_DENIED(200811,"没有开通权限"),
    PERMISSION_DENIED_SECRET(2006111,"服务中不存在该秘钥"),
    OTHER(2009111,"登录失败"),
    LOGIN_FAIL2(2102111,"IDM登录失败"),
    ;


    private Integer code;
    private String msg;

    IdmLoginCode(Integer code, String msg){
        this.code = code;
        this.msg = msg;
    }
}
LoginCode.LOGIN_SUCCESS.getCode()
@Getter
public enum DeletedEnum {
    /**
     * 状态
     */
    NORMAL(0,"正常"),
    /**
     * 禁用
     */
    FORBIDDEN(1,"禁用"),
    ;


    private Integer code;
    private String msg;

    DeletedEnum(Integer code, String msg){
        this.code = code;
        this.msg = msg;
    }
}
DeletedEnum.NORMAL.getCode()
https://blog.csdn.net/qq_27093465/article/details/52180865

用法一:常量

在JDK1.5 之前,我们定义常量都是: public static final… 。现在好了,有了枚举,可以把相关的常量分组到一个枚举类型里,而且枚举提供了比常量更多的方法。

Java代码

public enum Color {    RED, GREEN, BLANK, YELLOW  } 

用法二:switch

JDK1.6之前的switch语句只支持int,char,enum类型,使用枚举,能让我们的代码可读性更强。

Java代码

enum Signal {GREEN, YELLOW, RED}

public class TrafficLight {
    Signal color = Signal.RED;

    public void change() {
        switch (color) {
            case RED:
                color = Signal.GREEN;
                break;
            case YELLOW:
                color = Signal.RED;
                break;
            case GREEN:
                color = Signal.YELLOW;
                break;
        }
    }
}  

用法三:向枚举中添加新方法

如果打算自定义自己的方法,那么必须在enum实例序列的最后添加一个分号。

https://blog.csdn.net/qq_27093465/article/details/52180865

而且 Java 要求必须先定义 enum 实例。

Java代码

enum Color {
    RED("红色", 1), GREEN("绿色", 2), BLANK("白色", 3), YELLO("黄色", 4);
    // 成员变量     
    private String name;
    private int index;

    // 构造方法      
    private Color(String name, int index) {
        this.name = name;
        this.index = index;
    }

    // 普通方法     
    public static String getName(int index) {
        for (Color c : Color.values()) {
            if (c.getIndex() == index) {
                return c.name;
            }
        }
        return null;
    }

    // get set 方法
    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getIndex() {
        return index;
    }

    public void setIndex(int index) {
        this.index = index;
    }
}

用法四:覆盖枚举的方法

下面给出一个toString()方法覆盖的例子。

Java代码

public enum Color {
    RED("红色", 1), GREEN("绿色", 2), BLANK("白色", 3), YELLO("黄色", 4);
    // 成员变量      
    private String name;
    private int index;

    // 构造方法      
    private Color(String name, int index) {
        this.name = name;
        this.index = index;
    }      //覆盖方法      

    @Override
    public String toString() {
        return this.index + "_" + this.name;
    }
}  

用法五:实现接口

所有的枚举都继承自java.lang.Enum类。由于Java 不支持多继承,所以枚举对象不能再继承其他类。

Java代码

public interface Behaviour {
    void print();

    String getInfo();
}

public enum Color implements Behaviour {
    RED("红色", 1), GREEN("绿色", 2), BLANK("白色", 3), YELLO("黄色", 4);
    // 成员变量      
    private String name;
    private int index;

    // 构造方法     
    private Color(String name, int index) {
        this.name = name;
        this.index = index;
    }

    //接口方法      
    @Override
    public String getInfo() {
        return this.name;
    }

    //接口方法      
    @Override
    public void print() {
        System.out.println(this.index + ":" + this.name);
    }
}  

用法六:使用接口组织枚举

Java代码

public interface Food {
    enum Coffee implements Food {BLACK_COFFEE, DECAF_COFFEE, LATTE, CAPPUCCINO}

    enum Dessert implements Food {FRUIT, CAKE, GELATO}
}  
/**
 * 测试继承接口的枚举的使用(by 大师兄 or 大湿胸。)
 */
private static void testImplementsInterface(){
    for(Food.DessertEnum dessertEnum:Food.DessertEnum.values()){
        System.out.print(dessertEnum+"  ");
    }
    System.out.println();
    //我这地方这么写,是因为我在自己测试的时候,把这个coffee单独到一个文件去实现那个food接口,而不是在那个接口的内部。
    for(CoffeeEnum coffee:CoffeeEnum.values()){
        System.out.print(coffee+"  ");
    }
    System.out.println();
    //搞个实现接口,来组织枚举,简单讲,就是分类吧。如果大量使用枚举的话,这么干,在写代码的时候,就很方便调用啦。        //还有就是个“多态”的功能吧,
    Food food=Food.DessertEnum.CAKE;
    System.out.println(food);
    food=CoffeeEnum.BLACK_COFFEE;
    System.out.println(food);
}

运行结果

用法七:关于枚举集合的使用

java.util.EnumSet和java.util.EnumMap是两个枚举集合。EnumSet保证集合中的元素不重复;EnumMap中的 key是enum类型,而value则可以是任意类型。关于这个两个集合的使用就不在这里赘述,可以参考JDK文档。

关于枚举的实现细节和原理请参考:

参考资料:《ThinkingInJava》第四版

http://softbeta.iteye.com/blog/1185573

下面是我自己的测试代码。**

package com.lxk.enumTest; 
/**
 * Java枚举用法测试 * <p> * Created by lxk on 2016/12/15
 */
public class EnumTest {
    public static void main(String[] args) {
        forEnum();
        useEnumInJava();
    }

    /**
     * 循环枚举,输出ordinal属性;若枚举有内部属性,则也输出。(说的就是我定义的TYPE类型的枚举的typeName属性)
     */
    private static void forEnum() {
        for (SimpleEnum simpleEnum : SimpleEnum.values()) {
            System.out.println(simpleEnum + "  ordinal  " + simpleEnum.ordinal());
        }
        System.out.println("------------------");
        for (TYPE type : TYPE.values()) {
            System.out.println("type = " + type + "    type.name = " + type.name() + "   typeName = " + type.getTypeName() + "   ordinal = " + type.ordinal());
        }
    }

    /**
     * 在Java代码使用枚举
     */
    private static void useEnumInJava() {
        String typeName = "f5";
        TYPE type = TYPE.fromTypeName(typeName);
        if (TYPE.BALANCE.equals(type)) {
            System.out.println("根据字符串获得的枚举类型实例跟枚举常量一致");
        } else {
            System.out.println("大师兄代码错误");
        }
    }

    /**
     * 季节枚举(不带参数的枚举常量)这个是最简单的枚举使用实例     * Ordinal 属性,对应的就是排列顺序,从0开始。
     */
    private enum SimpleEnum {SPRING, SUMMER, AUTUMN, WINTER}

    /**
     * 常用类型(带参数的枚举常量,这个只是在书上不常见,实际使用还是很多的,看懂这个,使用就不是问题啦。)
     */
    private enum TYPE {
        FIREWALL("firewall"), SECRET("secretMac"), BALANCE("f5");
        private String typeName;

        TYPE(String typeName) {
            this.typeName = typeName;
        }

        /**
         * 根据类型的名称,返回类型的枚举实例。         *         * @param typeName 类型名称
         */
        public static TYPE fromTypeName(String typeName) {
            for (TYPE type : TYPE.values()) {
                if (type.getTypeName().equals(typeName)) {
                    return type;
                }
            }
            return null;
        }

        public String getTypeName() {
            return this.typeName;
        }
    }
}

然后是测试的结果图:

简单的例子,大家基本都用过,看不懂的基本都是第二个例子。可以看到,在第二个例子里面,后面带有参数,其实可以这么理解。

enum这个关键字,可以理解为跟class差不多,这也个单独的类。可以看到,上面的例子里面有属性,有构造方法,有getter,也可以有setter,但是一般都是构造传参数。还有其他自定义方法。那么在这些东西前面的,以逗号隔开的,最后以分号结尾的,这部分叫做,这个枚举的实例。也可以理解为,class new 出来的实例对象。这下就好理解了。只是,class,new对象,可以自己随便new,想几个就几个,而这个enum关键字,他就不行,他的实例对象,只能在这个enum里面体现。也就是说,他对应的实例是有限的。这也就是枚举的好处了,限制了某些东西的范围,举个栗子:一年四季,只能有春夏秋冬,你要是字符串表示的话,那就海了去了,但是,要用枚举类型的话,你在enum的大括号里面把所有的选项,全列出来,那么这个季节的属性,对应的值,只能在里面挑。不能有其他的。

我上面的例子,就是根据typeName,你可以从那些例子里面挑选到唯一的一个TYPE类型的枚举实例–TYPE.BALANCE。注意方法

TYPE type = TYPE.fromTypeName(typeName);
这个方法的返回类型就是这个TYPE枚举类型的。
这下就好理解,这个枚举是怎么在工作了吧

再补充一下:

上面那个带参数的枚举类型的实例里面实际上是三个属性,除了我自定义的typeName以外,还有2个是系统自带的。看下面源码的图:

看到这里之后,不知道你能不能理解下面图片内说明的话:下面图片主要说明在使用枚举时,的规范和标准。希望可以在实际开发时候用到

最后补充一点:

也许你知道呢,但是也许你不知道呢?我是真的不知道,测了之后才知道!!!

枚举类型对象之间的值比较,是可以使用==,直接来比较值,是否相等的,不是必须使用equals方法的哟。

具体,请参考下面的链接:

java 枚举类比较是用==还是equals?

2017.11.07 更新

有的老铁,说这个switch case怎么写,我就在下面再啰嗦一下。

    private static void testSwitchCase() {
        String typeName = "f5";
        //这几行注释呢,你可以试着三选一,测试一下效果。        
        //String typeName = "firewall";        
        // String typeName = "secretMac";       
        TypeEnum typeEnum = TypeEnum.fromTypeName(typeName);
        if (typeEnum == null) {
            return;
        }
        switch (typeEnum) {
            case FIREWALL:
                System.out.println("枚举名称(即默认自带的属性 name 的值)是:" + typeEnum.name());
                System.out.println("排序值(默认自带的属性 ordinal 的值)是:" + typeEnum.ordinal());
                System.out.println("枚举的自定义属性 typeName 的值是:" + typeEnum.getTypeName());
                break;
            case SECRET:
                System.out.println("枚举名称(即默认自带的属性 name 的值)是:" + typeEnum.name());
                System.out.println("排序值(默认自带的属性 ordinal 的值)是:" + typeEnum.ordinal());
                System.out.println("枚举的自定义属性 typeName 的值是:" + typeEnum.getTypeName());
                break;
            case BALANCE:
                System.out.println("枚举名称(即默认自带的属性 name 的值)是:" + typeEnum.name());
                System.out.println("排序值(默认自带的属性 ordinal 的值)是:" + typeEnum.ordinal());
                System.out.println("枚举的自定义属性 typeName 的值是:" + typeEnum.getTypeName());
                break;
            default:
                System.out.println("default");
        }
    }

然后,就是运行结果的截图。

老铁们,看完这个枚举,你要懂个概念,那就是,这个枚举,他是个对象,就像你定义的Student类,Person类,等等一些个类一样。

要有这么个概念。只要是个类,他就可以有构造函数,可以有属性,可以有方法。

对的,老铁,你对这个属性,构造函数啥的,有概念吧,没有的话,我可就郁闷啦。

然后,你就看到,这个地方有2个默认的属性,一个是name,一个是ordinal,这2个属性就像你定义Student类和Person类的name和age一样,

只不过,这2个是系统自带的属性,不用你自己去定义啦。

你也可以给这个枚举类,也就是你自己声明的枚举,随便加属性。

我上面代码例子里面的那个TypeEnum那个枚举,就是这么干的,就简单的添加了个自定义属性typeName,

虽然他有自己的name了,那姑且叫我这个自定义的属性叫别名吧。

可以看到,我例子里面就是通过自己写的那个构造方法给我这个自定义的属性初始化值的。

还有,这个构造方法是不可以,也不被运行public的,不信,你可以试试。

还有,你不能对系统自带的name属性,在构造函数里面赋值,没有为什么。

https://blog.csdn.net/qq_27093465/article/details/52180865

(2019-07-29更新)

再补充一点,这个也比较重要

就是这个枚举类型,一旦创建,且被使用(比如,存数据库啥的)之后,持久化后的对象信息里面就保存了这个枚举信息。这个时候你的需求或者要求啥的,需要变更这个枚举名称。应当禁止这个变更的操作,只能重新创建,用新的代替旧的,不能直接把旧的给改了,因为,就数据在逆转成对象的时候,如果,旧的枚举不在了,那么就会400还是500的报错或者是空指针的bug。这也是需要关注的一个问题。希望注意下,不然等到出bug了再想到这个问题,就不好了。

(2023-10-12更新)


package com.framework.modules.model.enums.extend;
import lombok.Getter;

public class DataSetEnums {

    /**
     * 数据集导入状态
     */
    @Getter
    public enum SetImportStatus {

        /**
         * 导入失败
         */
        FAIL(-1, "导入失败"),

        /**
         * 无图片
         */
        NONE(0, "无图片"),
        /**
         * 进行中
         */
        IMPORTING(1, "进行中"),
        /**
         * 已完成
         */
        FINISH(2, "已完成"),
        ;

        /**
         * 返回值
         */
        private final Integer code;
        /**
         * 返回信息
         */
        private final String msg;

        SetImportStatus(Integer code, String msg) {
            this.code = code;
            this.msg = msg;
        }
    }
    /**
     * 导出状态:0未导出 1导出
     */
    @Getter
    public enum ExportStatus {

        /**
         * 已导出
         */
        EXPORT(1, "已导出"),
        /**
         * 已导出
         */
        UN_EXPORT(0, "未导出"),

        ;
        /**
         * 返回值
         */
        private final Integer code;
        /**
         * 返回信息
         */
        private final String msg;

        ExportStatus(Integer code, String msg) {
            this.code = code;
            this.msg = msg;
        }
    }
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值