Java枚举(enum)详解

1. 枚举概述

枚举:列举,一个一个地列出来。
Java枚举:把某个类型的对象,全部列出来。

  • 什么情况下会用到枚举类型?

某些类的对象只有确定的有限个时,可以把这样的类声明为枚举。

例如:

星期:Monday(星期一)…Sunday(星期天)

性别:Male(男)、Female(女)

月份:January(1月)…December(12月)

季节:Spring(春天)…Winter(冬天)

  • 枚举是一种特殊的类,特殊在它的对象是有限的几个常量对象。它既是一种类(class)类型却又比类类型多了些特殊的约束,枚举的本质是一种受限制的类。

当需要定义一组常量时,可以使用枚举。

2. 枚举类型的定义

枚举是JDK1.5新增的引用数据类型。

  • 枚举的对象的属性不应允许被改动,所以应该使用 private final 修饰

    属性应该在构造器中为其赋值

    若枚举显式的定义了带参数的构造器,则在列出枚举值时也必须对应地传入参数

  • 枚举的实现:

    • jdk1.5之前,自定义枚举类
    • jdk1.5,可以使用 enum 关键字定义枚举

2.1 自定义枚举类

  • 要求:
    • 私有化类的构造器,保证不能在类的外部创建其对象
    • 在类的内部创建枚举类的实例,并用声明为 public static final 的变量来引用
    • 对象如果有实例变量,应该声明为private final,并在构造器中初始化
class Season {
    //声明Season对象的属性:使用 private final 修饰
    private final String seasonName;
    private final String seasonDesc;

    //私有化Season类的构造器,并给对象属性赋值
    private Season(String seasonName, String seasonDesc) {
        this.seasonName = seasonName;
        this.seasonDesc = seasonDesc;
    }

    //在类的内部创建当前枚举类的对象,变量用 public static final 修饰
    //public:外部可以使用枚举类的实例
    //static:外部可以使用“类名.对象名”调用
    //final:修饰的变量只能指向一个对象,保存的地址值不能更改
    public static final Season SPRING = new Season("春天", "春暖花开");
    public static final Season SUMMER = new Season("夏天", "烈日炎炎");
    public static final Season AUTUMN = new Season("秋天", "秋高气爽");
    public static final Season WINTER = new Season("冬天", "白雪皑皑");

    //其他需求
    //获取枚举类的对象的属性
    public String getSeasonName() {
        return seasonName;
    }

    public String getSeasonDesc() {
        return seasonDesc;
    }
    
    //提供toString()
    @Override
    public String toString() {
        return "Season{" +
                "seasonName='" + seasonName + '\'' +
                ", seasonDesc='" + seasonDesc + '\'' +
                '}';
    }
}

public class SeasonTest {
    public static void main(String[] args) {
        Season spring = Season.SPRING;
        System.out.println(spring.getSeasonName());
        System.out.println(spring);
    }
}

2.2 使用enum定义枚举

jdk1.5之前,枚举的定义格式太繁琐,存在很多重复代码
使用 enum 关键字简化枚举的定义

  • 声明格式:

    [修饰符] enmu 枚举名{
    	//实例列表
    	//其他成员列表
    }
    

枚举的所有实例必须在其他成员的前面显式列出(多个实例之间 , 分隔 ; 结尾,只存在实例列表时,最后一个实例后面的 ; 可省略,存在其他成员时,不能省略),它们实际上都是 public static final 修饰的常量对象,不用自己添加修饰符,填上反而会报错。

枚举的所有构造器只能使用 private 访问控制符,并且是隐式私有的。

switch 表达式的case 子句可以直接使用 Enum定义的枚举的对象的名字。

如果枚举中只有一个对象,则可以作为单例模式的一种实现方式。

//RED、GREEN、BLUE不是枚举Color的属性,而是它的对象
enum Color {
    //只有对象列表,可省略最后一个对象后的;
    //对象后面没有“(参数列表)”就是调用无参构造
    RED, GREEN, BLUE
}

//SPRING、SUMMER、AUTUMN、WINTER就是枚举Season的对象
enum Season {
    //对象列表必须写在类的前面,多个对象之间用,隔开,末尾对象;结束
    //对象后面有“(参数列表)”就是调用有参构造
    SPRING("春天", "春暖花开"),
    SUMMER("夏天", "烈日炎炎"),
    AUTUMN("秋天", "秋高气爽"),
    WINTER("冬天", "白雪皑皑"); 

    //声明Season对象的属性:使用 private final 修饰
    private final String seasonDesc;
    private final String seasonName;

    //私有化Season类的构造器,并给对象属性赋值
    private Season(String seasonName, String seasonDesc) {
        this.seasonName = seasonName;
        this.seasonDesc = seasonDesc;
    }

    //其他需求
    //获取枚举的对象的属性
    public String getSeasonName() {
        return seasonName;
    }

    public String getSeasonDesc() {
        return seasonDesc;
    }

    //一般不用重写toString
}

public class SeasonTest {
    public static void main(String[] args) {
        Color blue = Color.BLUE;
        System.out.println(blue); //BLUE
        System.out.println(Color.class.getSuperclass()); //class java.lang.Enum

        Season summer = Season.SUMMER;
        System.out.println(summer); //SUMMER
        System.out.println(Season.class.getSuperclass()); //class java.lang.Enum
        
        Season s = Season.SPRING;
        switch(s) {
            case SPRING:
                System.out.println("春暖花开");
                break;
            case SUMMER:
                System.out.println("烈日炎炎");
                break;
            case AUTUMN:
                System.out.println("秋高气爽");
                break;
            case WINTER:
                System.out.println("白雪皑皑");
                break;
        }
    }
}

3. 枚举的隐含直接父类:Enum类

枚举不能继承其他类型,因为枚举类型有一个隐含的直接父类 java.lang.Enum :它是所有枚举的父类。

public abstract class Enum<E extends Enum<E>>
        implements Comparable<E>, Serializable {...}

Enum类中有一个唯一的构造器:protected Enum(String name, int ordinal)

这个构造器不是程序员手动调用的,是编译器自动调用,在所有枚举类型的构造器的首行,并且自动传入name和ordinal的值。

name:就是枚举对象名称

ordinal:就是枚举对象的序号(在枚举声明中的位置),其中初始常量序号为0

3.1 Enum类的主要方法

除了toString方法,都是final修饰的方法,因此都不能重写。

toString():返回此枚举常量的名称,与其在枚举声明中的声明完全相同

String name():返回此枚举常量的名称,与其在枚举声明中的声明完全相同,一般用toString()

int ordinal():返回此枚举常量的序号(在枚举声明中的位置),其中初始常量序号为0

  • API中没有的方法,是编译器帮我们生成的方法:

values()方法:返回枚举类型的对象数组。该方法可以很方便地遍历所有的枚举值。

valueOf(String str):根据枚举常量的名称返回对应的枚举对象。要求字符串必须是枚举对象的“名字”。如不是,会有运行时异常

//便利枚举的对象
Season[] seasons = Season.values();
for (Season season : seasons) {
    System.out.println(season);
}
//SPRING
//SUMMER
//AUTUMN
//WINTER

Season spring = Season.valueOf("SPRING");
System.out.println(spring); //SPRING

4. 枚举实现接口

和普通 Java 类一样,枚举也可以实现一个或多个接口。

若每个枚举值在调用实现的接口方法呈现相同的行为,则只要统一实现该方法即可。

若需要每个枚举值在调用实现的接口方法呈现出不同的行为,则可以让每个枚举值分别来实现该方法。

定义接口:

interface MyInter {
   void message();
}

interface Info {
    void show();
}

枚举实现接口:

enum Season implements MyInter, Info {
    SPRING("春天", "春暖花开") {
        @Override
        public void show() {
            System.out.println("春天在哪里");
        }
    },
    SUMMER("夏天", "烈日炎炎") {
        @Override
        public void show() {
            System.out.println("夏天的风");
        }
    },
    AUTUMN("秋天", "秋高气爽") {
        @Override
        public void show() {
            System.out.println("秋天不回来");
        }
    },
    WINTER("冬天", "白雪皑皑") {
        @Override
        public void show() {
            System.out.println("大约在冬季");
        }
    };

    private final String seasonDesc;
    private final String seasonName;

    private Season(String seasonName, String seasonDesc) {
        this.seasonName = seasonName;
        this.seasonDesc = seasonDesc;
    }

    //统一实现的重写方法
    @Override
    public void message() {
        System.out.println("一年四季各不同");
    }
}

测试类:

public class SeasonTest {
    public static void main(String[] args) {
        Season spring = Season.SPRING;
        spring.message();
        Season summer = Season.SUMMER;
        summer.message();
        Season autumn = Season.AUTUMN;
        autumn.show();
        Season winter = Season.WINTER;
        winter.show();
    }
}
一年四季各不同
一年四季各不同
秋天不回来
大约在冬季

对于同一个方法,两种重写方式可任选其一,也可同时存在。分别重写时,所有的实例都要自己重写接口中的方法,否则必须添加枚举的统一重写方法。同时存在时,实例自己的重写方法会覆盖枚举的统一重写方法。

枚举既可以包含具体方法,也可以包含抽象方法。 如果枚举含有抽象方法,则枚举的每个实例都必须分别实现该方法。与实现接口类似。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值