1.枚举类的使用
当一个类的对象只有有限个,并且是确定的,我们就称这个类为"枚举类"。
举例如下:
星期 | Monday(星期一)、…、Sunday(星期天) |
---|---|
性别 | Man(男)、Woman(女) |
季节 | Spring(春节)…Winter(冬天) |
支付方式 | Cash(现金)、WeChatPay(微信)、Alipay(支付宝)、BankCard(银 行卡)、CreditCard(信用卡) |
就职状态 | Busy、Free、Vocation、Dimission |
订单状态 | Nonpayment(未付款)、Paid(已付款)、Delivered(已发货)、 Return(退货)、Checked(已确认)Fulfilled(已配货) |
线程状态 | 创建、就绪、运行、阻塞、死亡 |
(注:1. 当需要定义一组常量时,强烈建议使用枚举类
2. 如果枚举类中只有一个对象,则可以作为单例模式的实现方式。)
1.1 自定义枚举类(jdk5.0之前)
下面以季节为例做一下测试:
自定义Season枚举类(jdk5.0之前)
步骤:1. 私有化类的构造器,保证不能在类的外部创建其对象;
2. 在类的内部创建枚举类的实例。声明为:public static final;
3. 对象如果有实例变量(属性),应该声明为 private final,并在构造器中初始化;
设计Season枚举类如下:
package com.atguigu.java;
import com.sun.media.sound.SoftTuning;
/*
* 一、枚举类的使用
* 1.枚举类的理解:类的对象只有有限个,确定的,我们称此为枚举类
* 2.当需要定义一组常量时,强烈建议使用枚举类
* 3.如果枚举类中只有一个对象,则可以作为单例模式的实现方式。
* 二、如何定义枚举类
* 方式一:jdk5.0之前,自定义枚举类
* 方式二:jdk5.0,可以使用enum关键字定义枚举类。
*
* */
//自定义枚举类
class Season{
//1.声明Season对象的属性: private final修饰
private final String seasonName;
private final String seasonDesc;
//2.私有化类的构造器,并给对象属性赋值
private Season(String seasonName, String seasonDesc){
this.seasonName = seasonName;
this.seasonDesc = seasonDesc;
}
//3.提供当前枚举类的多个对象: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("冬天","冰天雪地");
//4.其他诉求:①获取枚举类对象的属性
public String getSeasonName() {
return seasonName;
}
public String getSeasonDesc() {
return seasonDesc;
}
//4.其他诉求:②提供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);
}
}
测试结果如下:
枚举类的属性:
- 枚举类对象的属性不应允许被改动, 所以应该使用 private final 修饰
- 枚举类的使用 private final 修饰的属性应该在构造器中为其赋值
- 若枚举类显式的定义了带参数的构造器, 则在列出枚举值时也必须对应的传入参数
1.2 使用enum定义枚举类
在jdk5.0及其之后,我们可以使用enum关键字定义枚举类。在使用过程中有如下的一些注意事项:
- 使用 enum 定义的枚举类默认继承了 java.lang.Enum类,因此不能再继承其他类;
- 枚举类的构造器只能使用 private 权限修饰符;
- 枚举类的所有实例必须在枚举类中显式列出(, 分隔 ; 结尾)。列出的实例系统会自动添加 public static final 修饰;(都是 public static final修饰的,所以就不用我们手动添加了,主要是为了省事儿)
- 必须在枚举类的第一行声明枚举类对象。
下面把之前的自定义枚举类Season改为使用enum的方式:
package com.atguigu.java;
/*
* 使用enum关键字定义枚举类
* 说明:定义的枚举类默认继承于java.lang.Enum类。
*
* */
//使用enum关键字定义枚举类
enum Season1{
//3.提供当前枚举类的多个对象,对象之间用","隔开,末尾对象以";"结束
SPRING("春天","春暖花开"),
SUMMER("夏天","夏日炎炎"),
AUTUMN("秋天","秋高气爽"),
WINTER("冬天","冰天雪地");
//1.声明Season对象的属性: private final修饰
private final String seasonName;
private final String seasonDesc;
//2.私有化类的构造器,并给对象属性赋值
private Season1(String seasonName, String seasonDesc){
this.seasonName = seasonName;
this.seasonDesc = seasonDesc;
}
//4.其他诉求:①获取枚举类对象的属性
public String getSeasonName() {
return seasonName;
}
public String getSeasonDesc() {
return seasonDesc;
}
}
public class Season1Test1 {
public static void main(String[] args) {
Season1 summer = Season1.SUMMER;
System.out.println(summer);
System.out.println(Season1.class.getSuperclass());
}
}
测试结果如下:
在枚举类Season1中我们没有提供toString()方法,但是在输出它的对象summer的时候打印的并不是地址值,可见Season1类并不是继承于Object类的;获取它的父类可以看到,它是继承于java.lang包下的enum类的,这就验证了第一条注意事项。
2.Enum类的主要方法
lang包下的Enum类提供了很多的方法,如下所示:
然而最常用的是下面的三个:
values()方法 | 返回枚举类型的对象数组。该方法可以很方便地遍历所有的 枚举值。 |
---|---|
valueOf(String str) | 可以把一个字符串转为对应的枚举类对象。要求字符 串必须是枚举类对象的“名字”。如不是,会有运行时异常: IllegalArgumentException。 |
toString() | 返回当前枚举类对象常量的名称 |
下面还拿Season1这个枚举类验证一下这三个方法:
Season1的设计不变:
//使用enum关键字定义枚举类
enum Season1{
//3.提供当前枚举类的多个对象,对象之间用","隔开,末尾对象以";"结束
SPRING("春天","春暖花开"),
SUMMER("夏天","夏日炎炎"),
AUTUMN("秋天","秋高气爽"),
WINTER("冬天","冰天雪地");
//1.声明Season对象的属性: private final修饰
private final String seasonName;
private final String seasonDesc;
//2.私有化类的构造器,并给对象属性赋值
private Season1(String seasonName, String seasonDesc){
this.seasonName = seasonName;
this.seasonDesc = seasonDesc;
}
//4.其他诉求:①获取枚举类对象的属性
public String getSeasonName() {
return seasonName;
}
public String getSeasonDesc() {
return seasonDesc;
}
}
Enum类的方法的测试:
public class Season1Test1 {
public static void main(String[] args) {
Season1 summer = Season1.SUMMER;
//toString():返回当前枚举类对象常量的名称
System.out.println(summer.toString());
System.out.println("*******************************");
//values():返回枚举类型的对象数组。该方法可以很方便地遍历所有的枚举值。
Season1[] values = Season1.values();
for(int i = 0; i < values.length; i++){
System.out.println(values[i]);
}
System.out.println("*******************************");
//再以线程的状态做一下测试
Thread.State[] values1 = Thread.State.values();
for (int i = 0; i < values1.length; i++) {
System.out.println(values1[i]);
}
//valueOf(String objName):根据提供的objName,返回枚举类中对象名为objName的对象
Season1 winter = Season1.valueOf("WINTER");
System.out.println(winter);
}
}
测试结果如下:
3.枚举类实现接口
和普通 Java 类一样,枚举类可以实现一个或多个接口 ,使用enum关键字定义的枚举类实现接口的情况有两类:
- 若每个枚举值在调用实现的接口方法呈现相同的行为方式,则只要统一实现该方法即可。
具体实现:实现接口,进而在enum类中实现抽象方法
//要实现的接口,里面声明了一个抽象方法show()
interface Info{
void show();
}
//使用enum关键字定义枚举类
enum Season1 implements Info{
//3.提供当前枚举类的多个对象,对象之间用","隔开,末尾对象以";"结束
SPRING("春天","春暖花开"),
SUMMER("夏天","夏日炎炎"),
AUTUMN("秋天","秋高气爽"),
WINTER("冬天","冰天雪地");
//1.声明Season对象的属性: private final修饰
private final String seasonName;
private final String seasonDesc;
//2.私有化类的构造器,并给对象属性赋值
private Season1(String seasonName, String seasonDesc){
this.seasonName = seasonName;
this.seasonDesc = seasonDesc;
}
//4.其他诉求:①获取枚举类对象的属性
public String getSeasonName() {
return seasonName;
}
public String getSeasonDesc() {
return seasonDesc;
}
//重写Info接口中的抽象方法show()
@Override
public void show(){
System.out.println("这是一个季节!");
}
}
给出如下的测试:
public class Season1Test1 {
public static void main(String[] args) {
Season1 summer = Season1.SUMMER;
summer.show();
}
}
测试结果如下:
那么使用这种方式实现接口的特点就是:不管我们是调用哪个季节的show方法,结果都是一样的。所以我们更希望的情况是能够根据不同的对象实现不同的效果,也就是类似多态的特性。
那么就要引入实现接口的第二种情况:
2. 若需要每个枚举值在调用实现的接口方法呈现出不同的行为方式, 则可以让每个枚举值分别来实现该方法。
那么这种方式下我们就需要让枚举类的对象分别去实现接口中的抽象方法。如下所示:
//要实现的接口,里面声明了一个抽象方法show()
interface Info{
void show();
}
//使用enum关键字定义枚举类
enum Season1 implements Info{
//3.提供当前枚举类的多个对象,对象之间用","隔开,末尾对象以";"结束
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("窗含西岭千秋雪");
}
};
//1.声明Season对象的属性: private final修饰
private final String seasonName;
private final String seasonDesc;
//2.私有化类的构造器,并给对象属性赋值
private Season1(String seasonName, String seasonDesc){
this.seasonName = seasonName;
this.seasonDesc = seasonDesc;
}
//4.其他诉求:①获取枚举类对象的属性
public String getSeasonName() {
return seasonName;
}
public String getSeasonDesc() {
return seasonDesc;
}
@Override
public void show(){
System.out.println("这是一个季节!");
}
}
可以看到,我们除了为每个对象实现了分别实现了接口中抽象方法外,也保留着Season1这个类中实现的抽象方法,主要是为了验证一下优先级的问题。
下面针对上述设计给出测试:
public class Season1Test1 {
public static void main(String[] args) {
//values():返回枚举类型的对象数组。该方法可以很方便地遍历所有的枚举值。
Season1[] values = Season1.values();
for(int i = 0; i < values.length; i++){
System.out.println(values[i]);
values[i].show(); //调用各个对象的方法
}
System.out.println("*******************************");
//valueOf(String objName):根据提供的objName,返回枚举类中对象名为objName的对象
Season1 winter = Season1.valueOf("WINTER");
winter.show();
}
}
测试结果如下:
可以看到,第二种实现方式的优先级更高。理解起来就是:更具体的实现会覆盖掉类中抽象的实现。