Java枚举类(enum)

Posted 爱敲代码的三毛

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java枚举类(enum)相关的知识,希望对你有一定的参考价值。


一、背景及定义

枚举是在JDK1.5以后引入的。主要用途是:将一组常量组织起来,在这之前表示一组常量通常使用定义常量的方式。

public static int final RED = 1;
public static int final GREEN = 2;
public static int final BLACK = 3;

优点:将常量组织起来统一进行管理
场景:错误状态码,消息类型,颜色的划分,状态机等等…

二、枚举的使用

1.switch

public enum  TestEnum {
    RED,BLACK,GREEN,WHITE;

    public static void main(String[] args) {
        TestEnum testEnum = TestEnum.WHITE;
        switch(testEnum) {
            case RED :
                System.out.println("red");
                break;
            case BLACK:
                System.out.println("black");
                break;
            case GREEN:
                System.out.println("green");
                break;
            case WHITE:
                System.out.println("white");
                break;
            default:
                System.out.println("enum");
                break;
        }
    }
}

运行结果:

2.常用方法

(1) values() 方法

以数组形式放回枚举类型的所有成员
代码示例:

public enum  TestEnum {
    RED,BLACK,GREEN,WHITE;

    public static void main(String[] args) {
        TestEnum[] testEnums = TestEnum.values();
        for (int i = 0; i < testEnums.length; i++) {
            System.out.println(testEnums[i]+"——>"+testEnums[i].ordinal());
        }
    }
}

运行结果:

(2) valueOf ()

该方法是把一个字符串转换为枚举成员
注意:这个枚举要存在不然会抛出异常;

public enum  TestEnum {
    RED,BLACK,GREEN,WHITE;

    public static void main(String[] args) {
        TestEnum testEnum1 = TestEnum.valueOf("RED");
        TestEnum testEnum2 = TestEnum.valueOf("BLACK");
        System.out.println(testEnum1);
        System.out.println(testEnum2);
    }
}

运行结果

(3) compareTo() 方法

该方法是比较两个枚举成员定义的位置,位置在前面的小于位置在后面的,会返回 一个小于0的数字。位置在后面的和位置在前面的比则会返回一个小于0的数字 .

public enum  TestEnum {
    RED,BLACK,GREEN,WHITE;

    public static void main(String[] args) {
        int tmp1 = TestEnum.RED.compareTo(TestEnum.BLACK);
        int tmp2 = TestEnum.WHITE.compareTo(TestEnum.GREEN);
        System.out.println(tmp1);
        System.out.println(tmp2);
    }
}

运行结果:

3.构造方法

注意:枚举的构造方法默认就是private的
所有的枚举类,都是默认继承与java.lang.Enum

如果改成 public 就会报错

三、枚举和反射

我们刚刚在反射里边看到了,任何一个类,哪怕其构造方法是私有的,我们也可以通过反射拿到他的实例对象,那么枚举的构造方法也是私有的,我们是否可以拿到呢?

import java.lang.reflect.Constructor;

public enum TestEnum {
    RED("red",0),BLACK("black",1),GREEN("green",2),WHITE("white",3);
    private String name;
    private int n;
    TestEnum(String name,int n) {
        this.name = name;
        this.n = n;
    }
    public static void reflectPrivateConstructor() {
        try {
            //1.获取Class对象
            Class<?> c = Class.forName("TestEnum");
            //2.获取构造方法
            Constructor<?> constructor = c.getDeclaredConstructor(String.class,int.class);
            //3.修改权限
            constructor.setAccessible(true);
            //4.获取对象示例
            TestEnum testEnum = (TestEnum)constructor.newInstance("蓝色",100);
            System.out.println(testEnum);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    public static void main(String[] args) {
        reflectPrivateConstructor();
    }
}

运行结果:

抛出异常:java.lang.NoSuchMethodException: TestEnum.(java.lang.String, int)

这个异常的意思是没有对应的构造方法,我们提供的枚举的构造方法就是两个参数分别是String 和 int ,问题出现在哪里呢?
所有的枚举类,都是默认继承于java.lang.Enum ,说到继承,继承了什么?继承了父类除构造函数外的所有东西,并且子类要帮助父类进行构造!
而我们写的类,并没有帮助父类构造!那意思是,我们要在自己的枚举类里面,提供super吗?不是的,枚举比较特殊,虽然我们写的是两个,但是默认他还添加了两个参数,哪两个参数呢?我们看一下Enum类的源码


也就是说,我们自己的构造函数有两个参数一个是String一个是int,同时他默认后边还会给两个参数,一个是String一个是int。也就是说,这里我们正确给的是4个参数:

public enum TestEnum {
    RED("red",0),BLACK("black",1),GREEN("green",2),WHITE("white",3);
    private String name;
    private int n;
    TestEnum(String name,int n) {
        this.name = name;
        this.n = n;
    }
    public static void reflectPrivateConstructor() {
        try {
            //1.获取Class对象
            Class<?> c = Class.forName("TestEnum");
            //2.获取构造方法
            Constructor<?> constructor = c.getDeclaredConstructor(String.class,int.class,String.class,int.class);
            //3.修改权限
            constructor.setAccessible(true);
            //4.获取对象示例
            TestEnum testEnum = (TestEnum)constructor.newInstance("蓝色",100,"父类参数",666);
            System.out.println(testEnum);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

运行结果:

这异常的意思是:无法创建反射枚举类
此时的异常信息显示,是我的一个方法这个方法是:newInstance() 报错了!


如果是枚举类直接就会抛出异常,所以不能通过反射创建枚举对象。


总结

  1. 枚举常量更简单安全 。
  2. 枚举具有内置方法 ,代码更优雅
  3. 枚举类是不能继承的
  4. 枚举的构造方法默认是私有的
  5. 枚举本身就是一个类,其构造方法默认为私有的,且都是默认继承与 java.lang.Enum

以上是关于Java枚举类(enum)的主要内容,如果未能解决你的问题,请参考以下文章

java中的enum

java枚举类型

java枚举类型

java枚举类Enum方法简介(valueof,value,ordinal)

java枚举类的几点疑惑

Java enum(枚举)使用详解之四