浅聊java中的枚举以及关于枚举的常见面试问题

2024-08-28 18:52

本文主要是介绍浅聊java中的枚举以及关于枚举的常见面试问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

基本概念:

什么是枚举?

枚举(Enum)是一种特殊的类,它用于表示一组固定的常量。枚举从Java 5(也称为Java 1.5)开始引入,提供了一种比传统常量(如使用public static final定义的变量)更类型安全、更易读和更易维护的方式来定义一组常量。

如何定义枚举?

枚举是通过关键字enum来定义的。枚举的每一个实例都是该枚举类型的一个对象,这些对象在编译时就已经被创建,并且不能被实例化。枚举类默认继承自java.lang.Enum,但不能显式继承其他类也不能被继承,因为当枚举被编译为类时会被final修饰,但它可以实现接口)。

public enum Color {  RED, GREEN, BLUE;  
}

枚举的特性有哪些?

  1. 类型安全:枚举提供了一种类型安全的方式来表示一组常量,避免了使用整数或字符串表示常量时可能发生的错误。
  2. 自动序列化:枚举实现了java.io.Serializable接口,因此它们可以自动序列化。
  3. 单例实现:枚举是实现单例模式的最佳方式之一,因为它保证了线程安全和实例的唯一性。
  4. 包含方法和属性:枚举可以包含构造方法(必须是私有的)、字段、方法和抽象方法。

枚举的常用方法有哪些?

  • values():返回包含枚举中所有元素的数组。
  • valueOf(String name):根据枚举常量的名称返回对应的枚举常量。
  • ordinal():返回枚举常量在枚举声明中的位置(从0开始)。
  • compareTo(E o):在枚举类型上实现自然排序。这个方法允许枚举常量按照它们在枚举声明中出现的顺序进行比较。
  • name():返回枚举常量的名称,这个名称就是声明枚举常量时所用的标识符。
  • toString():默认情况下,枚举的toString()方法返回枚举常量的名称。但你也可以重写这个方法以返回更有意义的字符串表示。

枚举的日常使用示例 :

定义

枚举类是通过enum关键字来定义的。枚举类的声明类似于类,但关键字是enum而不是class

public enum Day {  SUNDAY, MONDAY, TUESDAY, WEDNESDAY,  THURSDAY, FRIDAY, SATURDAY  
}

 使用

你可以像使用其他任何类型的变量一样使用枚举常量。枚举常量在编译时就被解析为对应的枚举类型对象。

Day day = Day.MONDAY;  
System.out.println(day); // 输出 MONDAY

方法

枚举可以包含字段、方法和构造函数。但是,枚举的构造函数默认是私有的,以防止外部代码实例化枚举。 

public enum Color {  RED("红色"), GREEN("绿色"), BLUE("蓝色");  private final String description;  Color(String description) {  this.description = description;  }  public String getDescription() {  return this.description;  }  
}  System.out.println(Color.RED.getDescription()); // 输出 红色
//
// 在这个例子中,Color枚举类有一个私有字段description和一个接受字符串参数的构造函数。每个枚举常量在声明时都通过调用构造函数来初始化其description字段。

遍历

你可以使用values()方法遍历枚举的所有值。 

for (Day d : Day.values()) {  System.out.println(d);  
}

枚举与switch

枚举与switch语句配合使用非常方便,因为switch语句可以接受枚举类型作为条件表达式。 

Day day = Day.MONDAY;  switch (day) {  case MONDAY:  System.out.println("星期一");  break;  case FRIDAY:  System.out.println("星期五");  break;  default:  System.out.println("其他日子");  
}

实现接口

枚举类可以实现一个或多个接口,并定义实现这些接口所需的方法。 

interface Describable {  String describe();  
}  public enum Planet implements Describable {  MERCURY, VENUS, EARTH;  @Override  public String describe() {  // 这里只是一个简单的实现,实际上你可以根据枚举常量的不同返回不同的描述  return this.name().toLowerCase() + " is a planet";  }  
}  System.out.println(Planet.EARTH.describe()); // 输出 earth is a planet

另一种写法:

 

interface Behaviour {  void show();  
}  public enum Animal implements Behaviour {  DOG {  @Override  public void show() {  System.out.println("Woof!");  }  },  CAT {  @Override  public void show() {  System.out.println("Meow!");  }  },  BIRD {  @Override  public void show() {  System.out.println("Tweet!");  }  };  // 示例:一个静态方法,遍历所有动物并显示它们的行为  public static void showAllAnimals() {  for (Animal animal : Animal.values()) {  animal.show();  }  }  // 测试方法  public static void main(String[] args) {  showAllAnimals();  }  
}

 带构造器和方法的枚举

public enum Season {  SPRING("Spring"), SUMMER("Summer"), AUTUMN("Autumn"), WINTER("Winter");  private final String description;  // 私有构造器  Season(String description) {  this.description = description;  }  // 公开方法  public String getDescription() {  return description;  }  // 示例方法,根据季节返回活动建议  public String getActivitySuggestion() {  switch (this) {  case SPRING:  return "Go for a hike or plant a garden.";  case SUMMER:  return "Go swimming or have a picnic.";  case AUTUMN:  return "Go apple picking or take a scenic drive.";  case WINTER:  return "Go skiing or build a snowman.";  default:  return "Enjoy the season!";  }  }  // 测试方法  public static void main(String[] args) {  Season season = Season.SUMMER;  System.out.println(season.getDescription() + ": " + season.getActivitySuggestion());  }  
}

 使用枚举作为Map的键

import java.util.EnumMap;  
import java.util.Map;  public enum Day {  MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY, SATURDAY, SUNDAY  
}  public class WorkHours {  public static void main(String[] args) {  Map<Day, Integer> workHours = new EnumMap<>(Day.class);  workHours.put(Day.MONDAY, 8);  workHours.put(Day.TUESDAY, 8);  workHours.put(Day.WEDNESDAY, 8);  workHours.put(Day.THURSDAY, 8);  workHours.put(Day.FRIDAY, 6); // 假设周五早些下班  // 周末不工作,所以不需要设置  for (Map.Entry<Day, Integer> entry : workHours.entrySet()) {  System.out.println(entry.getKey() + ": " + entry.getValue() + " hours");  }  }  
}

枚举中的抽象方法

 

public enum Operation {  PLUS {  @Override  public double apply(double x, double y) {  return x + y;  }  },  MINUS {  @Override  public double apply(double x, double y) {  return x - y;  }  },  MULTIPLY {  @Override  public double apply(double x, double y) {  return x * y;  }  },  DIVIDE {  @Override  public double apply(double x, double y) {  if (y == 0) throw new IllegalArgumentException("Cannot divide by zero");  return x / y;  }  };  // 抽象方法  public abstract double apply(double x, double y);  // 测试方法  public static void main(String[] args) {  System.out.println(Operation.PLUS.apply(10, 5));  // 输出 15.0  System.out.println(Operation.DIVIDE.apply(10, 2)); // 输出 5.0  }  
}

EnumSet与EnumMap:

EnumSet 和 EnumMap 是 Java 集合框架中专门为枚举(enum)类型设计的两个类。它们提供了比标准集合(如 HashSet 和 HashMap)更高的性能,特别是当集合中的元素是枚举类型时。这是因为它们内部利用了枚举的固有顺序和唯一性进行优化。

EnumSet

EnumSet 是一个不包含重复元素的集合,它专门为枚举类型设计。与普通的 Set 接口实现(如 HashSet)相比,EnumSet 在空间和时间上都有显著的优势,因为它内部使用位向量来表示集合中的元素,这种表示方法非常紧凑且高效。

主要特点

  • 类型安全:由于 EnumSet 的元素是枚举类型,因此在编译时就能保证类型安全。
  • 性能高效:内部使用位向量存储元素,因此集合的创建、遍历和查找等操作都非常快。
  • 丰富的构造函数EnumSet 提供了多种构造函数,包括从类型(Class<E>)创建空集合、从现有集合创建新集合等。
  • 迭代顺序EnumSet 遍历元素的顺序是枚举的自然顺序(除非指定了另一种顺序),这对于需要按特定顺序处理枚举值的场景非常有用。

示例代码

enum Day { MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY, SATURDAY, SUNDAY }  public class EnumSetExample {  public static void main(String[] args) {  EnumSet<Day> weekdays = EnumSet.range(Day.MONDAY, Day.FRIDAY);  for (Day day : weekdays) {  System.out.println(day);  }  }  
}

EnumMap

EnumMap 是一个映射(Map),其键是枚举类型。与普通的 Map 接口实现(如 HashMap)相比,EnumMap 提供了更高的性能,因为它内部使用数组来存储键值对并利用枚举的索引作为数组的下标

主要特点

  • 类型安全:由于 EnumMap 的键是枚举类型,因此在编译时就能保证类型安全。
  • 性能高效:内部使用数组存储键值对,使得查找、插入和删除操作都非常快。
  • 自然顺序EnumMap 遍历键的顺序是枚举的自然顺序(除非指定了另一种顺序)。
  • 内存紧凑:由于内部使用数组,且数组的大小固定为枚举类型的常量数,因此 EnumMap 的内存使用非常紧凑。

示例代码

enum Color { RED, GREEN, BLUE }  public class EnumMapExample {  public static void main(String[] args) {  EnumMap<Color, Integer> count = new EnumMap<>(Color.class);  count.put(Color.RED, 1);  count.put(Color.GREEN, 2);  count.put(Color.BLUE, 3);  for (Map.Entry<Color, Integer> entry : count.entrySet()) {  System.out.println(entry.getKey() + ": " + entry.getValue());  }  }  
}

总之,EnumSet 和 EnumMap 是 Java 集合框架中针对枚举类型优化的两个类,它们提供了类型安全、性能高效和内存紧凑的集合操作,是处理枚举类型数据时的首选集合类型。 

关于枚举的常见面试问题:

枚举如何实现单例模式?

  • 通过枚举的实例在JVM中唯一性,枚举天然就是单例的。例如,public enum Singleton { INSTANCE; }。这种方式实现单例模式既简单又安全,因为JVM保证了枚举实例的唯一性,并且枚举的构造方法是私有的,不能被外部实例化。

枚举可以继承其他类吗?

  • 枚举默认继承自java.lang.Enum类,因此不能显式继承其他类。

枚举的默认构造函数是怎样的?

  • 枚举的默认构造函数是私有的,不能显式定义访问修饰符(默认为private),且不能被外部实例化。

 枚举中的方法和属性是如何定义的?

  • 枚举中的方法和属性与普通类中的定义方式相同,但构造方法必须是私有的。

 values()valueOf(String name)方法的作用是什么?

  • values()方法返回包含枚举中所有元素的数组。
  • valueOf(String name)方法根据枚举常量的名称返回对应的枚举常量。

 枚举中的ordinal()方法有什么作用?

  • ordinal()方法返回枚举常量在枚举声明中的位置(从0开始)。

 为什么推荐使用枚举而不是静态常量?

  • 枚举提供了类型安全、自动序列化、单例实现等特性,而静态常量则不具备这些特性。此外,枚举的语义更清晰,易于理解和维护。

 这些问题涵盖了枚举的基本概念、特性、用法以及相关的设计模式,是Java枚举面试中常见的问题。在回答这些问题时,可以结合具体的代码示例和场景来解释,以增强回答的说服力和清晰度。

这篇关于浅聊java中的枚举以及关于枚举的常见面试问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1115603

相关文章

Python Jupyter Notebook导包报错问题及解决

《PythonJupyterNotebook导包报错问题及解决》在conda环境中安装包后,JupyterNotebook导入时出现ImportError,可能是由于包版本不对应或版本太高,解决方... 目录问题解决方法重新安装Jupyter NoteBook 更改Kernel总结问题在conda上安装了

pip install jupyterlab失败的原因问题及探索

《pipinstalljupyterlab失败的原因问题及探索》在学习Yolo模型时,尝试安装JupyterLab但遇到错误,错误提示缺少Rust和Cargo编译环境,因为pywinpty包需要它... 目录背景问题解决方案总结背景最近在学习Yolo模型,然后其中要下载jupyter(有点LSVmu像一个

如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解

《如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解》:本文主要介绍如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别的相关资料,描述了如何使用海康威视设备网络SD... 目录前言开发流程问题和解决方案dll库加载不到的问题老旧版本sdk不兼容的问题关键实现流程总结前言作为

SQL 中多表查询的常见连接方式详解

《SQL中多表查询的常见连接方式详解》本文介绍SQL中多表查询的常见连接方式,包括内连接(INNERJOIN)、左连接(LEFTJOIN)、右连接(RIGHTJOIN)、全外连接(FULLOUTER... 目录一、连接类型图表(ASCII 形式)二、前置代码(创建示例表)三、连接方式代码示例1. 内连接(I

解决jupyterLab打开后出现Config option `template_path`not recognized by `ExporterCollapsibleHeadings`问题

《解决jupyterLab打开后出现Configoption`template_path`notrecognizedby`ExporterCollapsibleHeadings`问题》在Ju... 目录jupyterLab打开后出现“templandroidate_path”相关问题这是 tensorflo

SpringBoot中使用 ThreadLocal 进行多线程上下文管理及注意事项小结

《SpringBoot中使用ThreadLocal进行多线程上下文管理及注意事项小结》本文详细介绍了ThreadLocal的原理、使用场景和示例代码,并在SpringBoot中使用ThreadLo... 目录前言技术积累1.什么是 ThreadLocal2. ThreadLocal 的原理2.1 线程隔离2

如何解决Pycharm编辑内容时有光标的问题

《如何解决Pycharm编辑内容时有光标的问题》文章介绍了如何在PyCharm中配置VimEmulator插件,包括检查插件是否已安装、下载插件以及安装IdeaVim插件的步骤... 目录Pycharm编辑内容时有光标1.如果Vim Emulator前面有对勾2.www.chinasem.cn如果tools工

Python安装时常见报错以及解决方案

《Python安装时常见报错以及解决方案》:本文主要介绍在安装Python、配置环境变量、使用pip以及运行Python脚本时常见的错误及其解决方案,文中介绍的非常详细,需要的朋友可以参考下... 目录一、安装 python 时常见报错及解决方案(一)安装包下载失败(二)权限不足二、配置环境变量时常见报错及

springboot将lib和jar分离的操作方法

《springboot将lib和jar分离的操作方法》本文介绍了如何通过优化pom.xml配置来减小SpringBoot项目的jar包大小,主要通过使用spring-boot-maven-plugin... 遇到一个问题,就是每次maven package或者maven install后target中的ja

Java中八大包装类举例详解(通俗易懂)

《Java中八大包装类举例详解(通俗易懂)》:本文主要介绍Java中的包装类,包括它们的作用、特点、用途以及如何进行装箱和拆箱,包装类还提供了许多实用方法,如转换、获取基本类型值、比较和类型检测,... 目录一、包装类(Wrapper Class)1、简要介绍2、包装类特点3、包装类用途二、装箱和拆箱1、装