学习Java——枚举

谁借莪1个温暖的怀抱¢ 2024-03-25 21:01 119阅读 0赞

目录

枚举的用法

定义

特点

应用场景

总结

用法

①、常量

②、switch

③、向枚举中添加新方法

④、覆盖枚举的方法

⑤、实现接口

⑥、使用接口组织枚举

每日寄语


枚举的用法

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

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

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

  1. private String getChineseSeason(int season){
  2. StringBuffer result = new StringBuffer();
  3. switch(season){
  4. case Season.SPRING :
  5. result.append("春天");
  6. break;
  7. case Season.SUMMER :
  8. result.append("夏天");
  9. break;
  10. case Season.AUTUMN :
  11. result.append("秋天");
  12. break;
  13. case Season.WINTER :
  14. result.append("冬天");
  15. break;
  16. default :
  17. result.append("地球没有的季节");
  18. break;
  19. }
  20. return result.toString();
  21. }
  22. public void doSomething(){
  23. System.out.println(this.getChineseSeason(Season.SPRING));//这是正常的场景
  24. System.out.println(this.getChineseSeason(5));//这个却是不正常的场景,这就导致了类型不安全问题
  25. }

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

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

定义

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

  1. public enum Season {
  2. SPRING, SUMMER, AUTUMN, WINTER;
  3. }

特点

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

  1. 使用关键字enum
  2. 类型名称,比如这里的Season
  3. 一串允许的值,比如上面定义的春夏秋冬四季
  4. 枚举可以单独定义在一个文件中,也可以嵌在其它Java类中。
  5. 枚举可以实现一个或多个接口(Interface)
  6. 可以定义新的变量
  7. 可以定义新的方法
  8. 可以定义根据具体枚举值而相异的类

应用场景

以在背景中提到的类型安全为例,用枚举类型重写那段代码。代码如下:

  1. public enum Season {
  2. SPRING(1), SUMMER(2), AUTUMN(3), WINTER(4);
  3. private int code;
  4. private Season(int code){
  5. this.code = code;
  6. }
  7. public int getCode(){
  8. return code;
  9. }
  10. }
  11. public class UseSeason {
  12. /**
  13. * 将英文的季节转换成中文季节
  14. * @param season
  15. * @return
  16. */
  17. public String getChineseSeason(Season season){
  18. StringBuffer result = new StringBuffer();
  19. switch(season){
  20. case SPRING :
  21. result.append("[中文:春天,枚举常量:" + season.name() + ",数据:" + season.getCode() + "]");
  22. break;
  23. case AUTUMN :
  24. result.append("[中文:秋天,枚举常量:" + season.name() + ",数据:" + season.getCode() + "]");
  25. break;
  26. case SUMMER :
  27. result.append("[中文:夏天,枚举常量:" + season.name() + ",数据:" + season.getCode() + "]");
  28. break;
  29. case WINTER :
  30. result.append("[中文:冬天,枚举常量:" + season.name() + ",数据:" + season.getCode() + "]");
  31. break;
  32. default :
  33. result.append("地球没有的季节 " + season.name());
  34. break;
  35. }
  36. return result.toString();
  37. }
  38. public void doSomething(){
  39. for(Season s : Season.values()){
  40. System.out.println(getChineseSeason(s));//这是正常的场景
  41. }
  42. //System.out.println(getChineseSeason(5));
  43. //此处已经是编译不通过了,这就保证了类型安全
  44. }
  45. public static void main(String[] arg){
  46. UseSeason useSeason = new UseSeason();
  47. useSeason.doSomething();
  48. }
  49. }

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

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

总结

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

用法

①、常量

  1. public enum Color {
  2. RED, GREEN, BLANK, YELLOW
  3. }

②、switch

  1. enum Signal {
  2. GREEN, YELLOW, RED
  3. }
  4. public class TrafficLight {
  5. Signal color = Signal.RED;
  6. public void change() {
  7. switch (color) {
  8. case RED:
  9. color = Signal.GREEN;
  10. break;
  11. case YELLOW:
  12. color = Signal.RED;
  13. break;
  14. case GREEN:
  15. color = Signal.YELLOW;
  16. break;
  17. }
  18. }
  19. }

③、向枚举中添加新方法

  1. public enum Color {
  2. RED("红色", 1), GREEN("绿色", 2), BLANK("白色", 3), YELLO("黄色", 4);
  3. // 成员变量
  4. private String name;
  5. private int index;
  6. // 构造方法
  7. private Color(String name, int index) {
  8. this.name = name;
  9. this.index = index;
  10. }
  11. // 普通方法
  12. public static String getName(int index) {
  13. for (Color c : Color.values()) {
  14. if (c.getIndex() == index) {
  15. return c.name;
  16. }
  17. }
  18. return null;
  19. }
  20. // get set 方法
  21. public String getName() {
  22. return name;
  23. }
  24. public void setName(String name) {
  25. this.name = name;
  26. }
  27. public int getIndex() {
  28. return index;
  29. }
  30. public void setIndex(int index) {
  31. this.index = index;
  32. }
  33. }

④、覆盖枚举的方法

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

⑤、实现接口

  1. public interface Behaviour {
  2. void print();
  3. String getInfo();
  4. }
  5. public enum Color implements Behaviour{
  6. RED("红色", 1), GREEN("绿色", 2), BLANK("白色", 3), YELLO("黄色", 4);
  7. // 成员变量
  8. private String name;
  9. private int index;
  10. // 构造方法
  11. private Color(String name, int index) {
  12. this.name = name;
  13. this.index = index;
  14. }
  15. //接口方法
  16. @Override
  17. public String getInfo() {
  18. return this.name;
  19. }
  20. //接口方法
  21. @Override
  22. public void print() {
  23. System.out.println(this.index+":"+this.name);
  24. }
  25. }

⑥、使用接口组织枚举

  1. public interface Food {
  2. enum Coffee implements Food{
  3. BLACK_COFFEE,DECAF_COFFEE,LATTE,CAPPUCCINO
  4. }
  5. enum Dessert implements Food{
  6. FRUIT, CAKE, GELATO
  7. }
  8. }

枚举的实现

Java SE5提供了一种新的类型-Java的枚举类型,关键字enum可以将一组具名的值的有限集合创建为一种新的类型,而这些具名的值可以作为常规的程序组件使用,这是一种非常有用的功能。

要想看源码,首先得有一个类吧,那么枚举类型到底是什么类呢?是enum吗?答案很明显不是,enum就和class一样,只是一个关键字,他并不是一个类,那么枚举是由什么类维护的呢,我们简单的写一个枚举:

  1. public enum t {
  2. SPRING,SUMMER;
  3. }

然后我们使用反编译,看看这段代码到底是怎么实现的,反编译后代码内容如下:

  1. public final class T extends Enum
  2. {
  3. private T(String s, int i)
  4. {
  5. super(s, i);
  6. }
  7. public static T[] values()
  8. {
  9. T at[];
  10. int i;
  11. T at1[];
  12. System.arraycopy(at = ENUM$VALUES, 0, at1 = new T[i = at.length], 0, i);
  13. return at1;
  14. }
  15. public static T valueOf(String s)
  16. {
  17. return (T)Enum.valueOf(demo/T, s);
  18. }
  19. public static final T SPRING;
  20. public static final T SUMMER;
  21. private static final T ENUM$VALUES[];
  22. static
  23. {
  24. SPRING = new T("SPRING", 0);
  25. SUMMER = new T("SUMMER", 1);
  26. ENUM$VALUES = (new T[] {
  27. SPRING, SUMMER
  28. });
  29. }
  30. }

通过反编译代码我们可以看到,public final class T extends Enum,说明,该类是继承了Enum类的,同时final关键字告诉我们,这个类也是不能被继承的。

当我们使用enmu来定义一个枚举类型的时候,编译器会自动帮我们创建一个final类型的类继承Enum类,所以枚举类型不能被继承。


每日寄语

把时间分给睡眠,分给书籍,分给运动,分给花鸟树木,还有山川湖水,而不是浪费在无用的人和事上,当你开始做时间的主人,哪些烦恼与不安也就自然烟消云散了 ——网友热评1a57c4f79f0841a9b31fa1228e78b371.jpeg

发表评论

表情:
评论列表 (有 0 条评论,119人围观)

还没有评论,来说两句吧...

相关阅读

    相关 Java进阶

    枚举进阶 上一节我们讲了[枚举初识][Link 1] 里面主要讲了枚举的实现原理,我们从编译器的角度看了枚举的底层实现以及枚举常用的方法 今天我们看一下枚举添加自定义方

    相关 IOS学习

               枚举就是自己定义的一些常量,而这些常量是根据自己的需要定义不同的用途。 有两种形式定义枚举。代码如下:          import <

    相关 学习笔记

    [2019独角兽企业重金招聘Python工程师标准>>> ][2019_Python_] ![hot3.png][] 枚举Enum > 关键字enum可以将一组具名的值