Java Review (十七、面向对象----枚举类)

简介: Java Review (十七、面向对象----枚举类)

文章目录

enum 的全称为 enumeration, 是 JDK 1.5 中引入的新特性。

在Java中,被 enum 关键字修饰的类型就是枚举类型。

枚举类入门

Java 5 新增了 一个 enum 关键宇 (它与 class 、 interface 关键字的地位相同),用以定义枚举类 。  枚举类是一种特殊的类,它一样可以有自己的成员变量、方法,可以实现一个或者多个接口,也可以定义自己的构造器 。一个 Java  源文件中最多只能定义一个 public 访问权限的枚举类,且该Java 源文件也必须和该枚举类的类名相同。

它与普通类有如下简单区别。

  • 枚举类可以实现一个或多个接口,使用 enum 定义的枚举类默认继承了 java.lang.Enum 类,而不是默认继承  Object 类,因此枚举类不能显式继承其他父类。其中 java.lang.Enum 类实现了java.lang.Serializable 和  java.lang.Comparable 两个接口 。
  • 使用 enum 定义、非抽象的枚举类默认会使用 final 修饰,因此枚举类不能派生子类 。
  • 枚举类的构造器只能使用 private 访问控制符 , 如果省略了构造器的访问控制符,则默认使用private 修饰 ; 如果强制指定访问控制符,则只能指定 private 修饰符 。
  • 枚举类的所有实例必须在枚举类的第一行显式列出,否则这个枚举类永远都不能产生实例  。 列出这些实例时,系统会自动添加 public static final 修饰,无须程序员显式添加 。枚举类默认提供了 一个  valuesO方法,该方法可以很方便地遍历所有的枚举值 。

下面程序定义了 一个 SeasonEnum 枚举类 。

SeasonEnum.java

public enum SeasonEnum {
  //在第一行列出 4 个枚举实例
  SPRING, SUMMER ,FALL,WINTER ;
}

定义枚举类时,需要显式列出所有的枚举值,如上面的SPRING,SUMMER,FALL,WINTER所示,所有的枚举值之间以英文逗号 , 隔开,枚举值列举结束后以英文分号作为结束 。 这些枚举值代表了该枚举类的所有可能的实例 。

如果需要使用该枚举类的某个实例,则可使用 EnumClass.variable 的形式,如 SeasonEnum. SPRING 。

EnumTest.java

public class EnumTest {
  public void judge(SeasonEnum s) {
    // switch 语句里的表达式可以是枚举值
    switch (s) {
      case SPRING:
        System.out . println( " 春暖花开,正好踏青") ;
        break ;
      case SUMMER :
        System . out.println( " 夏日炎炎,适合游泳 " ) ;
        break;
      case FALL:
       System.out.println( " 秋高气爽 , 进补及时 " ) ;
       break;
      case WINTER:
       System . out.println( " 冬日雪飘,国炉赏雪 " );
       break;
    }
  }  
  public static void main(String[] args){
   //枚举类默认有一个 values ()方法,返回该枚举类的所有实例
    for (SeasonEnum s : SeasonEnum.values()) {
      System.out.println(s) ;
    }
    //使用枚举实例时,可通过 EnumClass . variable 形式来访问
    new EnumTest().judge(SeasonEnum.SPRING) ;
  }
}

所有的枚举类都继承 了 java.lang.Enum 类,所以枚举类可以直接使用java .lang.Enum 类中所包含的方法 。 java . lang .Enum 类中提供了如下几个方法 :

image.png

枚举类的成员变量、方法和构造器

枚举类也是一种类,只是它是一种比较特殊的类,因此它一样可以定义成员变量、方法和构造器。

下面程序将定义一个 Gender 枚举类,该枚举类里包含了 一个 name 实例变量。

Gender.java

public enum Gender {
  MALE , FEMALE;
  //定义一个 public 修饰的实例变量
  public String name;
}

通过如下程序来使用该枚举类 。

GenderTest.java

public class GenderTest {
  public static void main(String[] args) {
    //通过 Enum 的 valueOf ()方法来获取指定枚举类的枚举值
    Gender g = Enum.valueOf(Gender.class , "FEMALE");
    //直接为枚举值的 name 实例变量赋傻
    g.name = "女";
    //直接访问枚举值的问me 实例变量
    System.out.println(g + "代表 :" + g.name);
  }
}

正如前面提到的, Java 应该把所有类设计成良好封装的类 ,所以不应该允许直接访问 Gender 类的name 成员 变量  ,而是应该通过方法来控制对 name 的访问 。 否则可能出现很混乱的情形,例如上面程序恰好设置了 g.name =“女”,要是采用  g.name = "男 ",那程序就会非常混乱了 ,可能出现 FEMALE 代表男的局面。可以按如下代码来改进 Gender 类的设计 。

Gender.java

public enum Gender {
  MALE , FEMALE ;
  //私有化,免其他程序直接访问该 name 成员变量
  private String name ;
  public void setName(String name){
   switch (this){
    case MALE :
     //MALE 枚举值的 name 变量则只能设置为"男 "
     if (name.equals("男")) {
       this.name = name;
     }else{
       System . out . println("参数错误") ;
       return;
     }  
    break ;
    case FEMALE:
     // FEMALE 枚举值的 name 变量只能设置为"女" 
     if (name.equals("女") ) {
      this.name = name;
     }else {
      System.out.println ("参数错误 ") ;
      return;
     } 
    break;
   }
  }
  public String getName() {
   return this.name;
  }
}

实现接口的枚举类

枚举类也可以实现一个或多个接口 。与普通类实现一个或多个接口完全一样 , 枚举类实现一个或多个接口时, 也需要实现该接口所包含的方法。下面程序定义了 一个 GenderDesc 接口。

GenderDesc.java

public interface GenderDesc{
  void info () ;
}  

在上面 GenderDesc 接口中定义了 一个 infoO方法,下面的 Gender 枚举类实现了该接口,并实现了该接口里包含的 info()方法 。 下面是 Gender 枚举类的代码 。

Gender.java

public enum Gender implements GenderDesc{
  MALE , FEMALE ;
  private String name ;
  public void setName(String name){
   switch (this){
    case MALE :
     if (name.equals("男")) {
       this.name = name;
     }else{
       System . out . println("参数错误") ;
       return;
     }  
    break ;
    case FEMALE:
     if (name.equals("女") ) {
      this.name = name;
     }else {
      System.out.println ("参数错误 ") ;
      return;
     } 
    break;
   }
  }
  public String getName() {
   return this.name;
  }
   //实现接口info()方法
   public void info(){
     System.out.println("这是一个用于定义性别的枚举类 " );
   }  
}

枚举类和普通类一样使用 implements 实现接口,井实现接口里包含的抽象方法 。

如果由枚举类来实现接口里的方法,则每个枚举值在调用该方法时都有相同的行为方式(因为方法体完全一样) 。

如果需要每个枚举值在调用该方法时呈现出不 同 的行为方式 , 则可以让每个枚举值分别来实现该方法 ,  每个枚举值提供不同的实现方式,从而让不同的枚举值调用该方法时具有不同的行为方式 。在下面的 Gender 枚举类中,不同的枚举值对  info()方法的实现各不相同 。

Gender.java

public enum Gender implements GenderDesc{
  //此处的枚举值必须调用对应的构造器来创建
    MALE ("男"){
    //花括号部分实际上是一个类体部分
    //并不是直接创建Gendar枚举类的实例
    //而是相当于创建 Gender 的匿名子类的实例
      public void info(){
        System.out.println("这个枚举值代表男性") ;
      } 
     }  
   FEMALE(" 女"){
     public void info(){
       System.out.println("这个枚举值代表女性") ;
     }
   }  
  private String name ;
  public void setName(String name){
   switch (this){
    case MALE :
     if (name.equals("男")) {
       this.name = name;
     }else{
       System.out.println("参数错误") ;
       return;
     }  
    break ;
    case FEMALE:
     if (name.equals("女") ) {
      this.name = name;
     }else {
      System.out.println ("参数错误 ") ;
      return;
     } 
    break;
   }
  }
  public String getName() {
   return this.name;
  } 
}

包含抽象方法的枚举类

假设有一个 Operation 枚举类,它的 4 个枚举值 PLUS, MINUS, TIMES, DIVIDE 分别代表加、减、乘、除 4 种运算,该枚举类需要定义一个 eval()方法来完成计算 。

从上面描述可以看出, Operation 需要让 PLUS 、 MINUS 、 TIMES 、 DIVIDE 四个值对  eval()方法各有不同的实现 。此时可考虑为 Operation 枚举类定义一个 evalO抽象方法,然后让 4 个枚举值分别为  eval()提供不同的实现 。 例如如下代码 。

Operation.java

public enum Operation {
   //定义每个枚举值时实现抽象方法
  PLUS{
    public double eval(double x , double y) {
      return x + y;
    }  
  },  
  MINUS{
    public double eval(double x , double y) {
      return x - y;
    }
  },    
  TIMES{
    public double eval(double x , double y) {
      return x * y;
    }
  } ,
  DIVIDE{  
    public double eval(double x , double y) {
      return x / y;
    }
  };  
  // 为枚举类定义一个抽象方法
  // 这个抽象方法由不同的枚举值提供不同的实现
  public abstract double eval(double x , double y);
  public static void main(String[] args) {
    System.out.println (Operation.PLUS.eval (3 , 4)) ;
    System.out.println(Operation.MINUS.eval(5,4)) ;
    System.out.println(Operation.TIMES.eval(5,4)) ;
    System.out.println(Operation.DIVIDE.eval(5,4));
  } 
}

编译上面程序会生成 5 个 class 文件,其中Operation 对应一个 class 文件,它的 4 个匿名内部子类,分别各对应一个 class 文件 。

枚举类里定义抽象方法时不能使用 abstract 关键字将枚举类定义成抽象类(因为系统自动会为它添加 abstract 关键宇),但因为枚举类需要显式创建枚举值,而不是作为父类,所以定义每个枚举值时必须为抽象方法提供实现,否则将出现编译错误 。


参考:

【1】:《疯狂Java讲义》

【2】:https://juejin.im/post/5c13133be51d456fac740c98


目录
相关文章
|
12天前
|
安全 Java 测试技术
🎉Java零基础:全面解析枚举的强大功能
【10月更文挑战第19天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
97 60
|
8天前
|
存储 安全 Java
java.util的Collections类
Collections 类位于 java.util 包下,提供了许多有用的对象和方法,来简化java中集合的创建、处理和多线程管理。掌握此类将非常有助于提升开发效率和维护代码的简洁性,同时对于程序的稳定性和安全性有大有帮助。
35 17
|
3天前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
28 4
|
4天前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
12 2
|
9天前
|
存储 安全 Java
如何保证 Java 类文件的安全性?
Java类文件的安全性可以通过多种方式保障,如使用数字签名验证类文件的完整性和来源,利用安全管理器和安全策略限制类文件的权限,以及通过加密技术保护类文件在传输过程中的安全。
|
13天前
|
Java 数据格式 索引
使用 Java 字节码工具检查类文件完整性的原理是什么
Java字节码工具通过解析和分析类文件的字节码,检查其结构和内容是否符合Java虚拟机规范,确保类文件的完整性和合法性,防止恶意代码或损坏的类文件影响程序运行。
|
13天前
|
Java API Maven
如何使用 Java 字节码工具检查类文件的完整性
本文介绍如何利用Java字节码工具来检测类文件的完整性和有效性,确保类文件未被篡改或损坏,适用于开发和维护阶段的代码质量控制。
|
11天前
|
Java 关系型数据库 数据库
面向对象设计原则在Java中的实现与案例分析
【10月更文挑战第25天】本文通过Java语言的具体实现和案例分析,详细介绍了面向对象设计的五大核心原则:单一职责原则、开闭原则、里氏替换原则、接口隔离原则和依赖倒置原则。这些原则帮助开发者构建更加灵活、可维护和可扩展的系统,不仅适用于Java,也适用于其他面向对象编程语言。
10 2
|
12天前
|
存储 Java 编译器
java wrapper是什么类
【10月更文挑战第16天】
20 3
|
15天前
|
Java 程序员 测试技术
Java|让 JUnit4 测试类自动注入 logger 和被测 Service
本文介绍如何通过自定义 IDEA 的 JUnit4 Test Class 模板,实现生成测试类时自动注入 logger 和被测 Service。
20 5
下一篇
无影云桌面