目录
什么是枚举
使用自定义枚举的步骤
自定义类实现枚举-小结
enum关键字实现枚举注意事项
enum练习
第一题
第二题
enum常用方法应用
enum实现接口细节
Enum习题
第一题
枚举简称(enmu)
枚举是一组常量的集合
可以理解成:枚举属于一种特殊的类,里面只包含一组有限的特点对象
实现枚举有两种方式 一种是自定义枚举 还有一种是使用枚举关键字
自定义类实现枚举
1.不需要提供setXxx方法,因为枚举对象值通常为只读.
2.对枚举对象/属性使用final+static共同修饰,实现底层优化.
3.枚举对象名通常使用全部大写,常量的命名规范.
4.枚举对象根据需要,也可以有多个属性
代码演示:
package idea.chapter11.enum_;
/**
* 演示自定义枚举的基本使用
*/
public class Enumeration02 {
public static void main(String[] args) {
System.out.println(Season.AUTUMN);
System.out.println(Season.SPRING);
}
}
//演示字定义枚举实现
class Season {//类
private String name;
private String desc;//描述
//定义了四个对象, 固定.
public static final Season SPRING = new Season("春天", "温暖");
public static final Season WINTER = new Season("冬天", "寒冷");
public static final Season AUTUMN = new Season("秋天", "凉爽");
public static final Season SUMMER = new Season("夏天", "炎热");
//1. 将构造器私有化,目的防止 直接 new
//2. 去掉setXxx方法, 防止属性被修改
//3. 在Season 内部,直接创建固定的对象
//4. 优化,可以加入 final 修饰符
private Season(String name, String desc) {
this.name = name;
this.desc = desc;
}
public String getName() {
return name;
}
public String getDesc() {
return desc;
}
@Override
public String toString() {
return "Season{" +
"name='" + name + '\'' +
", desc='" + desc + '\'' +
'}';
}
}
小结:进行自定义类实现枚举,有如下特点:
1)构造器私有化
2)本类内部创建一组对象
3)对外暴露对象(通过为对象添加public final static修饰符)
4)可以提供 get方法,但是不要提供 set
1.当我们使用enum关键字开发一个枚举类时,默认会继承Enum类,而且是一个final类[如何证明],使用javap工具来演示
2.传统的public static final Season2 SPRING =new Season2("春天”,"温暖 ");简化成SPRING("春天”,"温暖"),这里必须知道,它调用的是哪个构造器。主要看后面的括号如果有两个参数,说明调用的就是带两个参数的构造器
3.如果使用无参构造器 创建 枚举对象,则实参列表和小括号都可以省略
4.当有多个枚举对象时,使用,间隔,最后有一个分号结尾
5.枚举对象必须放在枚举类的行首.
代码演示:
package idea.chapter11.enum_;
public class Enumeration03 {
public static void main(String[] args) {
System.out.println(Season2.AUTUMN);
System.out.println(Season2.SUMMER);
/*
结论:
1.当我们使用enum关键字开发一个枚举类时,默认会继承Enum类,而且是一个final类[如何证明],使用javap工具来演示
2.传统的public static final Season2 SPRING =new Season2("春天”,"温暖 ");简化成SPRING("春天”,"温暖"),这里必须知道,它调用的是哪个构造器。主要看后面的括号如果有两个参数,说明调用的就是带两个参数的构造器
3.如果使用无参构造器 创建 枚举对象,则实参列表和小括号都可以省略
4.当有多个枚举对象时,使用,间隔,最后有一个分号结尾
5.枚举对象必须放在枚举类的行首.
*/
}
}
//演示使用enum关键字来实现枚举类
enum Season2 {//类
//定义了四个对象, 固定.
// public static final Season SPRING = new Season("春天", "温暖");
// public static final Season WINTER = new Season("冬天", "寒冷");
// public static final Season AUTUMN = new Season("秋天", "凉爽");
// public static final Season SUMMER = new Season("夏天", "炎热");
//两种方式的对比
//和之前的自定义枚举对比
//如果使用了enum 来实现枚举类
//1. 使用关键字 enum 替代 class
//2. public static final Season SPRING = new Season("春天", "温暖") 直接使用 SPRING("春天", "温暖")来替代 常量名(实参列表),这里的实参列表和我们类的构造器对应
//3. 如果有多个常量(对象), 使用 ,号间隔即可
//4. 如果使用enum 来实现枚举,要求将定义常量对象,写在前面,否则会报错,这是语法
//5. 如果我们使用的是无参构造器,创建常量对象,则可以省略 ()
//2.传统的public static final Season2 SPRING =new Season2("春天”,"温暖 ");简化成SPRING("春天”,"温暖"),这里必须知道,它调用的是哪个构造器。主要看后面的括号如果有两个参数,说明调用的就是带两个参数的构造器
//SPRING("春天", "温暖")这就表示调用的是Season2类的带两个参数的构造器
//5.枚举对象必须放在枚举类的行首.
SPRING("春天", "温暖"), WINTER("冬天", "寒冷"), AUTUMN("秋天", "凉爽"),
//3.如果使用无参构造器 创建 枚举对象,则实参列表和小括号都可以省略
SUMMER("夏天", "炎热")/*, What*/;//4.当有多个枚举对象时,使用,间隔,最后有一个分号结尾
private String name;
private String desc;//描述
private Season2() {//无参构造器
}
private Season2(String name, String desc) {
this.name = name;
this.desc = desc;
}
public String getName() {
return name;
}
public String getDesc() {
return desc;
}
@Override
public String toString() {
return "Season{" +
"name='" + name + '\'' +
", desc='" + desc + '\'' +
'}';
}
}
思路分析:
//这里其实就是调用Gender类的无参构造器
1)上面语法是ok
2)有一个枚举类Gender,没有属性。
3)有两个枚举对象 BOY,GIRL,使用的无参构造器创建.
下面代码是否正确,并说明表示的含义?
enum Gendert{
BOY,GIRL;
}
//这里其实就是调用Gender类的无参构造器
1)上面语法是ok
2)有一个枚举类Gender,没有属性。
3)有两个枚举对象 BOY,GIRL,使用的无参构造器创建.
思路分析: 1.在执行这两句话的时候都是没有问题 Gender2 boy = Gender2.BOY;//OK Gender2 boy2 = Gender2.BOY;//OK 2.紧接着就输出了boy 因为知道在输出对象实例的时候,会自动调用toString()方法,因为Gender2类是enum修饰的,会自动继承Enum类,并且调用父类的toString()方法 我们看Enum中的toString方法的源码 public String toString(){ return name; } 同过源码我们可以清楚的知道,返回的就是一个名字 所以在输出boy的时候,输出的就是BOY,输出的就是自己本身 System.out.println(boy2 == boy); //True 最后在判断boy2 和 boy 是否相同的时候,因为是enum修饰,会自动继承Enum类,并且在底层还会进行优化,是public static final 修饰的,因为是静态的所以访问的 都是同一个地方,因此返回true 并且Gender2.BOY 的本质是静态性质的,所以赋给boy 和 boy2本质是同一个对象
package idea.chapter11.enum_;
/**
* 演示枚举的习题
*/
public class EnumExercise01 {
public static void main(String[] args) {
Gender2 boy = Gender2.BOY;//OK
Gender2 boy2 = Gender2.BOY;//OK
System.out.println(boy);//输出BOY //本质就是调用 Gender2 的父类Enum的 toString()
//在输出boy时会默认的调用父类Enum的toString方法 改方法直接返回名字
//使用enum关键字默认会继承Java.lang.Enum 这个包
// public String toString() {
// return name;
// }
/*
思路分析:
1.在执行这两句话的时候都是没有问题
Gender2 boy = Gender2.BOY;//OK
Gender2 boy2 = Gender2.BOY;//OK
2.紧接着就输出了boy 因为知道在输出对象实例的时候,会自动调用toString()方法,因为Gender2类是enum修饰的,会自动继承Enum类,并且调用父类的toString()方法
我们看Enum中的toString方法的源码
public String toString(){
return name;
}
同过源码我们可以清楚的知道,返回的就是一个名字
所以在输出boy的时候,输出的就是BOY,输出的就是自己本身
System.out.println(boy2 == boy); //True
最后在判断boy2 和 boy 是否相同的时候,因为是enum修饰,会自动继承Enum类,并且在底层还会进行优化,是public static final 修饰的,因为是静态的所以访问的 都是同一个地方,因此返回true
并且Gender2.BOY 的本质是静态性质的,所以赋给boy 和 boy2本质是同一个对象
*/
System.out.println(boy2 == boy); //True
//因为是都是静态的属性 enum关键字在编译时属性的类型都是
// public static final 的 因为是静态所以指向的都是同一个地方所以返回true
}
}
enum Gender2{ //父类 Enum 的toString
BOY , GIRL;
}
toString:Enum类已经重写过了,返回的是当前对象名,子类可以重写该方法,用于返回对象的属性信息
name:返回当前对象名(常量名),子类中不能重写
ordinal:返回当前对象的位置号,默认从0开始
values:返回当前枚举类中所有的常量
valueOf:将字符串转换成枚举对象,要求字符串必须为已有的常量名,否则报异常!
compareTo比较两个枚举常量,比较的就是位置号
代码演示:
package idea.chapter11.enum_;
/**
* 演示Enum的各种方法
*/
public class EnumMethod {
public static void main(String[] args) {
Season2 autumn = Season2.AUTUMN;
//输出枚举对象的名字
System.out.println(autumn.name());
//ordinal() 输出的是该枚举对象的次序/编号,从0开始编号
//AUTUMN 枚举对象是第三个,因此输出 2
System.out.println(autumn.ordinal());
//从反编译可以看出 values方法,返回 Season2[]
//就是将枚举中的元素都放入到数组中
Season2[] values = Season2.values();
//增强for循环
for (Season2 season: values) {
System.out.println(season);
}
//valueOf:将字符串转换成枚举对象,要求字符串必须为已有的常量名,否则报异常
//执行流程
//1. 根据你输入的 "AUTUMN" 到 Season2的枚举对象去查找
//2. 如果找到了,就返回,如果没有找到,就报错
Season2 autumn1 = Season2.valueOf("AUTUMN");
System.out.println("autumn1=" + autumn);
//我们找到的autumn1和autumn是同一个对象,所以返回true
System.out.println(autumn == autumn1);
//compareTo:比较两个枚举常量,比较的就是编号
//1. 就是把 Season2.AUTUMN 枚举对象的编号 和 Season2.SUMMER枚举对象的编号比较
//2. 看看结果
/*
public final int compareTo(E o) {
return self.ordinal - other.ordinal;
}
Season2.AUTUMN的编号[2] - Season2.SUMMER的编号[3]
*/
System.out.println(Season2.AUTUMN.compareTo(Season2.SUMMER));
}
}
1)使用enum关键字后,就不能再继承其它类了,因为enum会隐式继承Enum,而Java是单继承机制。
2)枚举类和普通类一样,可以实现接口,如下形式。 enum 类名 implements接口1,接口2)
代码演示:
package idea.chapter11.enum_;
/**
* 演示接口的细节
*/
public class EnumDetail {
public static void main(String[] args) {
/*
1)使用enum关键字后,就不能再继承其它类了,因为enum会隐式继承Enum,而Java是单继承机制。
2)枚举类和普通类一样,可以实现接口,如下形式。
enum 类名 implements接口1,接口2)
*/
}
}
//因为 enum会隐式的继承Enum类,因为Java是单继承机制,所有enum修饰的不能在继承其他类,但是可以实现接口
enum Animal implements eat {
//DOG是一个枚举对象 没有参数默认调用无参构造器 枚举默认是public static final 修饰的所以可以通过类名直接调用
DOG;
@Override
public void eat() {
System.out.println("吃东西");
}
}
interface eat {
void eat();
}
代码演示:
package idea.chapter11.enum_;
public class EnumExercise02 {
public static void main(String[] args) {
/*
enum课堂练习
课堂完成EnumExercise02.java
1)声明Week枚举类,其中包含星期一至星期日的定义; ===所有星期的信息如下==MONDAY, TUESDAY, WEDNESDAY, THURSDAY, 星期一
FRIDAY, SATURDAY, SUNDAY; 星期二
2)使用values返回所有的枚举数组,并遍历,输出左图
*/
Week[] values = Week.values();
System.out.println("===所有星期如下===");
for (Week value : values) {
System.out.println(value);
}
}
}
enum Week {
MONDAY("星期一"), TUESDAY("星期二"), WEDNESDAY("星期三"), THURSDAY("星期四"), FRIDAY("星期五"), SATURDAY("星期六"), SUNDAY("星期日");
private String name;
Week(String name) {
this.name = name;
}
public String getName() {
return name;
}
@Override
public String toString() {
return name;
}
}