Java的枚举类型使用方法详解_java

1.背景
在java语言中还没有引入枚举类型之前,表示枚举类型的常用模式是声明一组具有int常量。之前我们通常利用public final static 方法定义的代码如下,分别用1 表示春天,2表示夏天,3表示秋天,4表示冬天。

public class Season {
 public static final int SPRING = 1;
 public static final int SUMMER = 2;
 public static final int AUTUMN = 3;
 public static final int WINTER = 4;
}

这种方法称作int枚举模式。可这种模式有什么问题呢,我们都用了那么久了,应该没问题的。通常我们写出来的代码都会考虑它的安全性、易用性和可读性。 首先我们来考虑一下它的类型安全性。当然这种模式不是类型安全的。比如说我们设计一个函数,要求传入春夏秋冬的某个值。但是使用int类型,我们无法保证传入的值为合法。代码如下所示:

private String getChineseSeason(int season){
  StringBuffer result = new StringBuffer();
  switch(season){
   case Season.SPRING :
    result.append("春天");
    break;
   case Season.SUMMER :
    result.append("夏天");
    break;
   case Season.AUTUMN :
    result.append("秋天");
    break;
   case Season.WINTER :
    result.append("冬天");
    break;
   default :
    result.append("地球没有的季节");
    break;
  }
  return result.toString();
 }

 public void doSomething(){
  System.out.println(this.getChineseSeason(Season.SPRING));//这是正常的场景

  System.out.println(this.getChineseSeason(5));//这个却是不正常的场景,这就导致了类型不安全问题
 }

程序getChineseSeason(Season.SPRING)是我们预期的使用方法。可getChineseSeason(5)显然就不是了,而且编译很通过,在运行时会出现什么情况,我们就不得而知了。这显然就不符合Java程序的类型安全。

接下来我们来考虑一下这种模式的可读性。使用枚举的大多数场合,我都需要方便得到枚举类型的字符串表达式。如果将int枚举常量打印出来,我们所见到的就是一组数字,这是没什么太大的用处。我们可能会想到使用String常量代替int常量。虽然它为这些常量提供了可打印的字符串,但是它会导致性能问题,因为它依赖于字符串的比较操作,所以这种模式也是我们不期望的。 从类型安全性和程序可读性两方面考虑,int和String枚举模式的缺点就显露出来了。幸运的是,从Java1.5发行版本开始,就提出了另一种可以替代的解决方案,可以避免int和String枚举模式的缺点,并提供了许多额外的好处。那就是枚举类型(enum type)。接下来的章节将介绍枚举类型的定义、特征、应用场景和优缺点。

2.定义
枚举类型(enum type)是指由一组固定的常量组成合法的类型。Java中由关键字enum来定义一个枚举类型。下面就是java枚举类型的定义。

public enum Season {
 SPRING, SUMMER, AUTUMN, WINER;
}

3.特点
Java定义枚举类型的语句很简约。它有以下特点:

1) 使用关键字enum 2) 类型名称,比如这里的Season 3) 一串允许的值,比如上面定义的春夏秋冬四季 4) 枚举可以单独定义在一个文件中,也可以嵌在其它Java类中。
除了这样的基本要求外,用户还有一些其他选择

5) 枚举可以实现一个或多个接口(Interface) 6) 可以定义新的变量 7) 可以定义新的方法 8) 可以定义根据具体枚举值而相异的类
4.应用场景
以在背景中提到的类型安全为例,用枚举类型重写那段代码。代码如下:

public enum Season {
 SPRING(1), SUMMER(2), AUTUMN(3), WINTER(4);

 private int code;
 private Season(int code){
  this.code = code;
 }

 public int getCode(){
  return code;
 }
}
public class UseSeason {
 /**
  * 将英文的季节转换成中文季节
  * @param season
  * @return
  */
 public String getChineseSeason(Season season){
  StringBuffer result = new StringBuffer();
  switch(season){
   case SPRING :
    result.append("[中文:春天,枚举常量:" + season.name() + ",数据:" + season.getCode() + "]");
    break;
   case AUTUMN :
    result.append("[中文:秋天,枚举常量:" + season.name() + ",数据:" + season.getCode() + "]");
    break;
   case SUMMER :
    result.append("[中文:夏天,枚举常量:" + season.name() + ",数据:" + season.getCode() + "]");
    break;
   case WINTER :
    result.append("[中文:冬天,枚举常量:" + season.name() + ",数据:" + season.getCode() + "]");
    break;
   default :
    result.append("地球没有的季节 " + season.name());
    break;
  }
  return result.toString();
 }

 public void doSomething(){
  for(Season s : Season.values()){
   System.out.println(getChineseSeason(s));//这是正常的场景
  }
  //System.out.println(getChineseSeason(5));
  //此处已经是编译不通过了,这就保证了类型安全
 }

 public static void main(String[] arg){
  UseSeason useSeason = new UseSeason();
  useSeason.doSomething();
 }
}

[中文:春天,枚举常量:SPRING,数据:1] [中文:夏天,枚举常量:SUMMER,数据:2] [中文:秋天,枚举常量:AUTUMN,数据:3] [中文:冬天,枚举常量:WINTER,数据:4]

这里有一个问题,为什么我要将域添加到枚举类型中呢?目的是想将数据与它的常量关联起来。如1代表春天,2代表夏天。

5.总结
那么什么时候应该使用枚举呢?每当需要一组固定的常量的时候,如一周的天数、一年四季等。或者是在我们编译前就知道其包含的所有值的集合。Java 1.5的枚举能满足绝大部分程序员的要求的,它的简明,易用的特点是很突出的。

6.用法
用法一:常量

public enum Color {
 RED, GREEN, BLANK, YELLOW
}

用法二:switch

enum Signal {
 GREEN, YELLOW, RED
}
public class TrafficLight {
 Signal color = Signal.RED;
 public void change() {
  switch (color) {
  case RED:
   color = Signal.GREEN;
   break;
  case YELLOW:
   color = Signal.RED;
   break;
  case GREEN:
   color = Signal.YELLOW;
   break;
  }
 }
}

用法三:向枚举中添加新方法

public enum Color {
 RED("红色", 1), GREEN("绿色", 2), BLANK("白色", 3), YELLO("黄色", 4);
 // 成员变量
 private String name;
 private int index;
 // 构造方法
 private Color(String name, int index) {
  this.name = name;
  this.index = index;
 }
 // 普通方法
 public static String getName(int index) {
  for (Color c : Color.values()) {
   if (c.getIndex() == index) {
    return c.name;
   }
  }
  return null;
 }
 // get set 方法
 public String getName() {
  return name;
 }
 public void setName(String name) {
  this.name = name;
 }
 public int getIndex() {
  return index;
 }
 public void setIndex(int index) {
  this.index = index;
 }
}

用法四:覆盖枚举的方法

public enum Color {
 RED("红色", 1), GREEN("绿色", 2), BLANK("白色", 3), YELLO("黄色", 4);
 // 成员变量
 private String name;
 private int index;
 // 构造方法
 private Color(String name, int index) {
  this.name = name;
  this.index = index;
 }
 //覆盖方法
 @Override
 public String toString() {
  return this.index+"_"+this.name;
 }
}

用法五:实现接口

public interface Behaviour {
 void print();
 String getInfo();
}
public enum Color implements Behaviour{
 RED("红色", 1), GREEN("绿色", 2), BLANK("白色", 3), YELLO("黄色", 4);
 // 成员变量
 private String name;
 private int index;
 // 构造方法
 private Color(String name, int index) {
  this.name = name;
  this.index = index;
 }
//接口方法
 @Override
 public String getInfo() {
  return this.name;
 }
 //接口方法
 @Override
 public void print() {
  System.out.println(this.index+":"+this.name);
 }
}

用法六:使用接口组织枚举

public interface Food {
 enum Coffee implements Food{
  BLACK_COFFEE,DECAF_COFFEE,LATTE,CAPPUCCINO
 }
 enum Dessert implements Food{
  FRUIT, CAKE, GELATO
 }
}

以上就是本文的全部内容,希望对大家的学习有所帮助。

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索java
枚举
java 枚举类型详解、枚举类型详解、java 枚举详解、java中的枚举详解、java枚举类型的用法,以便于您获取更多的相关知识。

时间: 2024-12-04 01:31:19

Java的枚举类型使用方法详解_java的相关文章

java中set接口使用方法详解_java

java中的set接口有如下的特点: 不允许出现重复元素: 集合中的元素位置无顺序: 有且只有一个值为null的元素. 因为java中的set接口模仿了数学上的set抽象,所以,对应的数学上set的特性为: 互异性:一个集合中,任何两个元素都认为是不相同的,即每个元素只能出现一次.无序性:一个集合中,每个元素的地位都是相同的,元素之间是无序的.集合上可以定义序关系,定义了序关系后,元素之间就可以按照序关系排序.但就集合本身的特性而言,元素之间没有必然的序.空集的性质:空集是一切集合的子集    

JAVA获取CLASSPATH路径的方法详解_java

ClassLoader 提供了两个方法用于从装载的类路径中取得资源: public URL getResource (String name); public InputStream getResourceAsStream (String name);        这里name是资源的类路径,它是相对与"/"根路径下的位置.getResource得到的是一个URL对象来定位资源,而getResourceAsStream取得该资源输入流的引用保证程序可以从正确的位置抽取数据.     

java.util.ArrayDeque类使用方法详解_java

本文为大家介绍了java.util.ArrayDeque类使用方法,供大家参考,具体内容如下 1. ArrayDeque有两个类属性,head和tail,两个指针. 2. ArrayDeque通过一个数组作为载体,其中的数组元素在add等方法执行时不移动,发生变化的只是head和tail指针,而且指针是循环变化,数组容量不限制. 3. offer方法和add方法都是通过其中的addLast方法实现,每添加一个元素,就把元素加到数组的尾部,此时,head指针没有变化,而tail指针加一,因为指针是

基于序列化存取实现java对象深度克隆的方法详解_java

我们知道,在java中,将一个非原型类型类型的对象引用,赋值给另一个对象的引用之后,这两个引用就指向了同一个对象,如: 复制代码 代码如下: public class DeepCloneTest {  private class CloneTest {  private Long myLong = new Long(1); }  public static void main(String args[]) {  new DeepCloneTest().Test(); }  public void

基于Java内存溢出的解决方法详解_java

一.内存溢出类型1.java.lang.OutOfMemoryError: PermGen spaceJVM管理两种类型的内存,堆和非堆.堆是给开发人员用的上面说的就是,是在JVM启动时创建:非堆是留给JVM自己用的,用来存放类的信息的.它和堆不同,运行期内GC不会释放空间.如果web app用了大量的第三方jar或者应用有太多的class文件而恰好MaxPermSize设置较小,超出了也会导致这块内存的占用过多造成溢出,或者tomcat热部署时侯不会清理前面加载的环境,只会将context更改

高阶Java-Java枚举类型enum应用详解

java的Enum枚举类型终于在j2se1.5出现了.之前觉得它只不过是鸡肋而已,可有可无.毕竟这么多年来,没有它,大家不都过得很好吗?今日看<Thinking in Java>4th edition,里面有一句话"有时恰恰因为它,你才能够"优雅而干净"地解决问题.优雅与清晰很重要,正式它们区别了成功的解决方案与失败的解决方案.而失败的解决方案就是因为其他人无法理他."使用Enum枚举类型,可以将以前笨拙的代码变得优雅简单?但是,我同时也在思考另外一个问

C# 为String类型增加方法详解_C#教程

namespace MyExtensionMethods { public static class MyExtensions { public static int MyGetLength(this System.String target) { return target.Length; } } } 使用时,需要引入这个名字空间,引用如下: string str = "dafasdf"; int len = str.MyGetLength(); 以上这篇C# 为String类型增加

Java Spring MVC 上传下载文件配置及controller方法详解_java

下载: 1.在spring-mvc中配置(用于100M以下的文件下载) <bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter"> <property name="messageConverters"> <list> <!--配置下载返回类型--> <bean class="or

Java的JSON处理器fastjson使用方法详解_java

fastjson 是一个性能很好的 Java 语言实现的 JSON 解析器和生成器,来自阿里巴巴的工程师开发. 主要特点: • 快速FAST (比其它任何基于Java的解析器和生成器更快,包括jackson) • 强大(支持普通JDK类包括任意Java Bean Class.Collection.Map.Date或enum) • 零依赖(没有依赖其它任何类库除了JDK) 示例代码: import com.alibaba.fastjson.JSON; Group group = new Group