浅聊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

相关文章

springboot健康检查监控全过程

《springboot健康检查监控全过程》文章介绍了SpringBoot如何使用Actuator和Micrometer进行健康检查和监控,通过配置和自定义健康指示器,开发者可以实时监控应用组件的状态,... 目录1. 引言重要性2. 配置Spring Boot ActuatorSpring Boot Act

使用Java解析JSON数据并提取特定字段的实现步骤(以提取mailNo为例)

《使用Java解析JSON数据并提取特定字段的实现步骤(以提取mailNo为例)》在现代软件开发中,处理JSON数据是一项非常常见的任务,无论是从API接口获取数据,还是将数据存储为JSON格式,解析... 目录1. 背景介绍1.1 jsON简介1.2 实际案例2. 准备工作2.1 环境搭建2.1.1 添加

Java实现任务管理器性能网络监控数据的方法详解

《Java实现任务管理器性能网络监控数据的方法详解》在现代操作系统中,任务管理器是一个非常重要的工具,用于监控和管理计算机的运行状态,包括CPU使用率、内存占用等,对于开发者和系统管理员来说,了解这些... 目录引言一、背景知识二、准备工作1. Maven依赖2. Gradle依赖三、代码实现四、代码详解五

Redis连接失败:客户端IP不在白名单中的问题分析与解决方案

《Redis连接失败:客户端IP不在白名单中的问题分析与解决方案》在现代分布式系统中,Redis作为一种高性能的内存数据库,被广泛应用于缓存、消息队列、会话存储等场景,然而,在实际使用过程中,我们可能... 目录一、问题背景二、错误分析1. 错误信息解读2. 根本原因三、解决方案1. 将客户端IP添加到Re

java如何分布式锁实现和选型

《java如何分布式锁实现和选型》文章介绍了分布式锁的重要性以及在分布式系统中常见的问题和需求,它详细阐述了如何使用分布式锁来确保数据的一致性和系统的高可用性,文章还提供了基于数据库、Redis和Zo... 目录引言:分布式锁的重要性与分布式系统中的常见问题和需求分布式锁的重要性分布式系统中常见的问题和需求

SpringBoot基于MyBatis-Plus实现Lambda Query查询的示例代码

《SpringBoot基于MyBatis-Plus实现LambdaQuery查询的示例代码》MyBatis-Plus是MyBatis的增强工具,简化了数据库操作,并提高了开发效率,它提供了多种查询方... 目录引言基础环境配置依赖配置(Maven)application.yml 配置表结构设计demo_st

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys

Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单

《Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单》:本文主要介绍Springboot的ThreadPoolTaskScheduler线... 目录ThreadPoolTaskScheduler线程池实现15分钟不操作自动取消订单概要1,创建订单后

详谈redis跟数据库的数据同步问题

《详谈redis跟数据库的数据同步问题》文章讨论了在Redis和数据库数据一致性问题上的解决方案,主要比较了先更新Redis缓存再更新数据库和先更新数据库再更新Redis缓存两种方案,文章指出,删除R... 目录一、Redis 数据库数据一致性的解决方案1.1、更新Redis缓存、删除Redis缓存的区别二

oracle数据库索引失效的问题及解决

《oracle数据库索引失效的问题及解决》本文总结了在Oracle数据库中索引失效的一些常见场景,包括使用isnull、isnotnull、!=、、、函数处理、like前置%查询以及范围索引和等值索引... 目录oracle数据库索引失效问题场景环境索引失效情况及验证结论一结论二结论三结论四结论五总结ora