有的人说,不推荐使用枚举。有的人说,枚举很好用。究竟怎么使用,如何使用,仁者见仁智者见智。总之,先学会再说~

为什么要引入枚举类

一个小案例

你写了一个小程序,不过好久不用了,突然有一天,你想使用一下它。程序要想正确运行,需要将今天星期几存到数据库里。这个时候,你开始犯难了。

当初的你还很年轻,不懂程序界的险恶,设计这个程序的时候,傻不拉几把这个字段设计为int类型的,用0代表周日,1代表周一。。。6代表周六,添加的时候就setWeekday(0)。但是这么长时间没用了,你忘记自己是从周一开始计算还是周日开始计算了,换句话说,你想不起来0代表的是周一还是周日了!

于是你各种翻代码,看数据库的字段,数据库保存的信息,终于搞懂了,你很开心,用了一次之后,觉得这个程序没意思,又不用了。

很久之后,你心血来潮,又想用一次它,很不幸,你又忘记到底0代表周一还是周日了,一番查找之后。你决定重构代码,因为你受不了了!!

静态变量来帮忙

经过一番思考,你决定使用七个静态变量来代表星期几,以后只要引用和静态变量就可以了,而不用自己输入012….你这么写:

public class Weekday {public final static int SUN = 0;public final static int MON = 1;public final static int TUE = 2;public final static int WED = 3;public final static int THU = 4;public final static int FRI = 5;public final static int SAT = 6;}

机智如你,这个时候,只要Weekday.SUN就可以了,不用操心到底应该填写0还是填写1。

但是这个时候的你,也不是当初初出茅庐的小伙子了,很明显,这样写已经不能满足你了。你还想让这个类做更多的事,比如,你想知道下一天是星期几,还想把今天是星期几打印出来。一番深思熟虑后,你改成了这样:

public class Weekday {private Weekday(){}public final static Weekday SUN = new Weekday();public final static Weekday MON = new Weekday();public final static Weekday TUE = new Weekday();public final static Weekday WED = new Weekday();public final static Weekday THU = new Weekday();public final static Weekday FRI = new Weekday();public final static Weekday SAT = new Weekday();public static Weekday  getNextDay(Weekday nowDay){if(nowDay == SUN) {return MON;}else if(nowDay == MON) {return TUE;}else if(nowDay == TUE) {return WED;}else if(nowDay == WED) {return THU;}else if(nowDay == THU) {return FRI;}else if(nowDay == FRI) {return SAT;}else {return SUN;}}public static void printNowDay(Weekday nowDay){if(nowDay == SUN)System.out.println("sunday");else if(nowDay == MON)System.out.println("monday");else if(nowDay == TUE)System.out.println("tuesday");else if(nowDay == WED)System.out.println("wednesday");else if(nowDay == THU)System.out.println("thursday");else if(nowDay == FRI)System.out.println("friday");elseSystem.out.println("saturday");}}class Test1{public static void main(String[] args) {Weekday nowday = Weekday.SUN;Weekday.printNowDay(nowday);Weekday nextDay = Weekday.getNextDay(nowday);System.out.print("nextday ====> ");Weekday.printNowDay(nextDay);}
}
//测试结果:
//sunday
//nextday ====> monday

哟,不错。考虑的很详细。并且私有构造方法后,外界就不能创建该类的对象了,这样就避免了星期八星期九的出现,所有Weekday的对象都在该类内部创建。

不对,好像缺了点什么,我要的是int!我的int呢?!。所以,你还需要一个这样的方法:

    public static int toInt(Weekday nowDay){if(nowDay == SUN)return 0;else if(nowDay == MON)return 1;else if(nowDay == TUE)return 2;else if(nowDay == WED)return 3;else if(nowDay == THU)return 4;else if(nowDay == FRI)return 5;elsereturn 6;}

当你需要一个整形数据的时候,只需要Weekday.toInt(Weekday.SUN);,看起来你好像完成了你的任务。

但是,你有没有发现,这样写,好麻烦啊。如果想要扩展一下功能,大量的ifelse会让人眼花缭乱。

有没有更好的方式呢?你大概已经知道了,没错,我们需要枚举类!

我们先来看看枚举类是什么。

一个简单的枚举类

话不多说,先来代码:

public enum Weekday {SUN,MON,TUS,WED,THU,FRI,SAT
}

代码这么少?

没错,这就是枚举类,我们来看看怎么使用它:

class Test2{public static void main(String[] args) {Weekday sun = Weekday.SUN;System.out.println(sun); // 输出 SUN}
}

看起来和上面的静态变量使用方式差不多,而且默认的toString方法返回的就是对应的名字。

我们上面的那段代码重写toString也是不可以打印出当前是星期几的,因为toString方法没有参数。所以我们自己写了一个printNowDay方法。

当然,这么简单的枚举类是不可能实现我们的要求的,所以,我们还要接着写:

public enum Weekday {SUN(0),MON(1),TUS(2),WED(3),THU(4),FRI(5),SAT(6);private int value;private Weekday(int value){this.value = value;}public static Weekday getNextDay(Weekday nowDay){int nextDayValue = nowDay.value;if (++nextDayValue == 7){nextDayValue =0;}return getWeekdayByValue(nextDayValue);}public static Weekday getWeekdayByValue(int value) {for (Weekday c : Weekday.values()) {if (c.value == value) {return c;}}return null;}
}class Test2{public static void main(String[] args) {System.out.println("nowday ====> " + Weekday.SAT);System.out.println("nowday int ====> " + Weekday.SAT.ordinal());System.out.println("nextday ====> " + Weekday.getNextDay(Weekday.SAT)); // 输出 SUN//输出://nowday ====> SAT//nowday int ====> 6//nextday ====> SUN}
}

这样就完成了我们的目标,和之前的代码比起来,有没有觉得突然高大上了许多?没有那么多烦人的ifelse,世界都清净了。

好了,现在你大概知道为什么要引入枚举类了吧?就是因为在没有枚举类的时候,我们要定义一个有限的序列,比如星期几,男人女人,春夏秋冬,一般会通过上面那种静态变量的形式,但是使用那样的形式如果需要一些其他的功能,需要些很多奇奇怪怪的代码。所以,枚举类的出现,就是为了简化这种操作。

可以将枚举类理解为是java的一种语法糖。

枚举类的用法

最简单的使用

最简单的枚举类就像我们上面第一个定义的枚举类一样:

public enum Weekday {SUN,MON,TUS,WED,THU,FRI,SAT
}

如何使用它呢?

先来看看它有哪些方法:

这是Weekday可以调用的方法和参数。发现它有两个方法:value()和valueOf()。还有我们刚刚定义的七个变量。

这些事枚举变量的方法。我们接下来会演示几个比较重要的:

public enum Weekday {SUN,MON,TUS,WED,THU,FRI,SAT
}class Test3{public static void main(String[] args) {System.out.println(Weekday.valueOf("mon".toUpperCase()));//MONfor (Weekday w : Weekday.values()){System.out.println(w + ".ordinal()  ====>" +w.ordinal());}//SUN.ordinal()  ====>0//MON.ordinal()  ====>1//TUS.ordinal()  ====>2//WED.ordinal()  ====>3//THU.ordinal()  ====>4//FRI.ordinal()  ====>5//SAT.ordinal()  ====>6System.out.println("Weekday.MON.compareTo(Weekday.FRI) ===> " + Weekday.MON.compareTo(Weekday.FRI));System.out.println("Weekday.MON.compareTo(Weekday.MON) ===> " + Weekday.MON.compareTo(Weekday.MON));System.out.println("Weekday.MON.compareTo(Weekday.SUM) ===> " + Weekday.MON.compareTo(Weekday.SUN));//Weekday.MON.compareTo(Weekday.FRI) ===> -4//Weekday.MON.compareTo(Weekday.MON) ===> 0//Weekday.MON.compareTo(Weekday.SUM) ===> 1System.out.println("Weekday.MON.name() ====> " + Weekday.MON.name());//Weekday.MON.name() ====> MON}
}

这段代码,我们演示了几个常用的方法和功能:

  1. Weekday.valueOf() 方法:

    它的作用是传来一个字符串,然后将它转变为对应的枚举变量。前提是你传的字符串和定义枚举变量的字符串一抹一样,区分大小写。如果你传了一个不存在的字符串,那么会抛出异常。

  2. Weekday.values()方法。

    这个方法会返回包括所有枚举变量的数组。在该例中,返回的就是包含了七个星期的Weekday[]。可以方便的用来做循环。

  3. 枚举变量的toString()方法。

    该方法直接返回枚举定义枚举变量的字符串,比如MON就返回【”MON”】。

  4. 枚举变量的.ordinal()方法。

    默认请款下,枚举类会给所有的枚举变量一个默认的次序,该次序从0开始,类似于数组的下标。而.ordinal()方法就是获取这个次序(或者说下标)

  5. 枚举变量的compareTo()方法。

    该方法用来比较两个枚举变量的”大小”,实际上比较的是两个枚举变量的次序,返回两个次序相减后的结果,如果为负数,就证明变量1”小于”变量2 (变量1.compareTo(变量2),返回【变量1.ordinal() - 变量2.ordinal()】)

    这是compareTo的源码,会先判断是不是同一个枚举类的变量,然后再返回差值。

  6. 枚举类的name()方法。

    它和toString()方法的返回值一样,事实上,这两个方法本来就是一样的:

    这两个方法的默认实现是一样的,唯一的区别是,你可以重写toString方法。name变量就是枚举变量的字符串形式。

还有一些其他的方法我就暂时不介绍了,感兴趣的话可以自己去看看文档或者源码,都挺简单的。

要点:

  • 使用的是enum关键字而不是class。
  • 多个枚举变量直接用逗号隔开。
  • 枚举变量最好大写,多个单词之间使用”_”隔开(比如:INT_SUM)。
  • 定义完所有的变量后,以分号结束,如果只有枚举变量,而没有自定义变量,分号可以省略(例如上面的代码就忽略了分号)。
  • 在其他类中使用enum变量的时候,只需要【类名.变量名】就可以了,和使用静态变量一样。

但是这种简单的使用显然不能体现出枚举的强大,我们来学习一下复杂的使用:

枚举的高级使用方法

就像我们前面的案例一样,你需要让每一个星期几对应到一个整数,比如星期天对应0。上面讲到了,枚举类在定义的时候会自动为每个变量添加一个顺序,从0开始。

假如你希望0代表星期天,1代表周一。。。并且你在定义枚举类的时候,顺序也是这个顺序,那你可以不用定义新的变量,就像这样:

public enum Weekday {SUN,MON,TUS,WED,THU,FRI,SAT
}

这个时候,星期天对应的ordinal值就是0,周一对应的就是1,满足你的要求。但是,如果你这么写,那就有问题了:

public enum Weekday {MON,TUS,WED,THU,FRI,SAT,SUN
}

我吧SUN放到了最后,但是我还是希0代表SUN,1代表MON怎么办呢?默认的ordinal是指望不上了,因为它只会傻傻的给第一个变量0,给第二个1。。。

所以,我们需要自己定义变量!

看代码:

public enum Weekday {MON(1),TUS(2),WED(3),THU(4),FRI(5),SAT(6),SUN(0);private int value;private Weekday(int value){this.value = value;}
}

我们对上面的代码做了一些改变:

首先,我们在每个枚举变量的后面加上了一个括号,里面是我们希望它代表的数字。

然后,我们定义了一个int变量,然后通过构造函数初始化这个变量。

你应该也清楚了,括号里的数字,其实就是我们定义的那个int变量。这句叫做自定义变量。

请注意:这里有三点需要注意:

  1. 一定要把枚举变量的定义放在第一行,并且以分号结尾。
  2. 构造函数必须私有化。事实上,private是多余的,你完全没有必要写,因为它默认并强制是private,如果你要写,也只能写private,写public是不能通过编译的。
  3. 自定义变量与默认的ordinal属性并不冲突,ordinal还是按照它的规则给每个枚举变量按顺序赋值。

好了,你很聪明,你已经掌握了上面的知识,你想,既然能自定义一个变量,能不能自定义两个呢?

当然可以:

public enum Weekday {MON(1,"mon"),TUS(2,"tus"),WED(3,"wed"),THU(4,"thu"),FRI(5,"fri"),SAT(6,"sat"),SUN(0,"sun");private int value;private String label;private Weekday(int value,String label){this.value = value;this.label = label;}
}

你可以定义任何你想要的变量。学完了这些,大概枚举类你也应该掌握了,但是,还有没有其他用法呢?

枚举类中的抽象类

如果我在枚举类中定义一个抽象方法会怎么样?

你要知道,枚举类不能继承其他类,也不能被其他类继承。至于为什么,我们后面会说到。

你应该知道,有抽象方法的类必然是抽象类,抽象类就需要子类继承它然后实现它的抽象方法,但是呢,枚举类不能被继承。。你是不是有点乱?

我们先来看代码:

public enum TrafficLamp {RED(30) {@Overridepublic TrafficLamp getNextLamp() {return GREEN;}}, GREEN(45) {@Overridepublic TrafficLamp getNextLamp() {return YELLOW;}}, YELLOW(5) {@Overridepublic TrafficLamp getNextLamp() {return RED;}};private int time;private TrafficLamp(int time) {this.time = time;}//一个抽象方法public abstract TrafficLamp getNextLamp();}

你好像懂了点什么。但是你好像又不太懂。为什么一个变量的后边可以带一个代码块并且实现抽象方法呢?

别着急,带着这个疑问,我们来看一下枚举类的实现原理。

枚举类的实现原理

从最简单的看起:

public enum Weekday {SUN,MON,TUS,WED,THU,FRI,SAT
}

还是这段熟悉的代码,我们编译一下它,再反编译一下看看它到底是什么样子的:

你是不是觉得很熟悉?反编译出来的代码和我们一开始用静态变量自己写的那个类出奇的相似!

而且,你看到了熟悉的values()方法和valueOf()方法。

仔细看,这个类继承了java.lang.Enum类!所以说,枚举类不能再继承其他类了,因为默认已经继承了Enum类。

并且,这个类是final的!所以它不能被继承!

回到我们刚才的那个疑问:

RED(30) {@Overridepublic TrafficLamp getNextLamp() {return GREEN;}
}

为什么会有这么神奇的代码?现在你差不多懂了。因为RED本身就是一个TrafficLamp对象的引用。实际上,在初始化这个枚举类的时候,你可以理解为执行的是TrafficLamp RED = new TrafficLamp(30) ,但是因为TrafficLamp里面有抽象方法,还记得匿名内部类么?

我们可以这样来创建一个TrafficLamp引用:

TrafficLamp RED = new TrafficLamp(30){@Overridepublic TrafficLamp getNextLamp() {return GREEN;}
};

而在枚举类中,我们只需要像上面那样写【RED(30){}】就可以了,因为java会自动的去帮我们完成这一系列操作。

如果你还是不太理解,那么你可以自己去反编译一下TrafficLamp这个类,看看jvm是怎么处理它的就明白了。

枚举类的其他用法

说一说枚举类的其他用法。

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 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;}// 接口方法@Overridepublic String getInfo() {return this.name;}// 接口方法@Overridepublic 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}
}

使用枚举创建单例模式

使用枚举创建的单例模式:

public enum EasySingleton{INSTANCE;
}

代码就这么简单,你可以使用EasySingleton.INSTANCE调用它,比起你在单例中调用getInstance()方法容易多了。

我们来看看正常情况下是怎样创建单例模式的:

用双检索实现单例:

下面的代码是用双检索实现单例模式的例子,在这里getInstance()方法检查了两次来判断INSTANCE是否为null,这就是为什么叫双检索的原因,记住双检索在java5之前是有问题的,但是java5在内存模型中有了volatile变量之后就没问题了。

public class DoubleCheckedLockingSingleton{private volatile DoubleCheckedLockingSingleton INSTANCE;private DoubleCheckedLockingSingleton(){}public DoubleCheckedLockingSingleton getInstance(){if(INSTANCE == null){synchronized(DoubleCheckedLockingSingleton.class){//double checking Singleton instanceif(INSTANCE == null){INSTANCE = new DoubleCheckedLockingSingleton();}}}return INSTANCE;}
}

你可以访问DoubleCheckedLockingSingleTon.getInstance()来获得实例对象。

用静态工厂方法实现单例:

public class Singleton{private static final Singleton INSTANCE = new Singleton();private Singleton(){}public static Singleton getSingleton(){return INSTANCE;}
}

你可以调用Singleton.getInstance()方法来获得实例对象。


上面的两种方式就是懒汉式和恶汉式单利的创建,但是无论哪一种,都不如枚举来的方便。而且传统的单例模式的另外一个问题是一旦你实现了serializable接口,他们就不再是单例的了。但是枚举类的父类【Enum类】实现了Serializable接口,也就是说,所有的枚举类都是可以实现序列化的,这也是一个优点。

总结

最后总结一下:

  • 可以创建一个enum类,把它看做一个普通的类。除了它不能继承其他类了。(java是单继承,它已经继承了Enum),可以添加其他方法,覆盖它本身的方法
  • switch()参数可以使用enum
  • values()方法是编译器插入到enum定义中的static方法,所以,当你将enum实例向上转型为父类Enum是,values()就不可访问了。解决办法:在Class中有一个getEnumConstants()方法,所以即便Enum接口中没有values()方法,我们仍然可以通过Class对象取得所有的enum实例
  • 无法从enum继承子类,如果需要扩展enum中的元素,在一个接口的内部,创建实现该接口的枚举,以此将元素进行分组。达到将枚举元素进行分组。
  • enum允许程序员为eunm实例编写方法。所以可以为每个enum实例赋予各自不同的行为。

本文到这里就差不多结束了。可能举得例子不是很恰当,代码写的不是很优雅,不过我只是用来引出枚举的,大家不要鸡蛋里头挑骨头哈哈。

如果文章内容有什么问题,请及时与我联系。

除此之外,还有两个枚举集合:【java.util.EnumSet和java.util.EnumMap】没有讲。关于枚举集合的使用会在后面讲集合框架的时候再详细讲解。


转载请注明出处:
本文地址:http://blog.csdn.net/qq_31655965/article/details/55049192
原创自csdn:http://blog.csdn.net/qq_31655965
博主:clever_fan

看完了,如果对你有帮助,随手点个赞呗~~~

参考资料:
http://www.cnblogs.com/happyPawpaw/archive/2013/04/09/3009553.html
https://my.oschina.net/zhoujy/blog/134958

重新认识java(十) ---- Enum(枚举类)相关推荐

  1. c枚举类型enum例题_一篇文章让你详细了解Java中Enum枚举类的使用

    文章前记 程序员工作久了便可能整日忙碌于"增删改查"中,迷失方向,毫无进步. 该公众号致力于分享软件开发相关的原创干货,助你完成从程序员到架构师的进阶之路! 努力!做一个NB的Co ...

  2. Java中的enum枚举类

    首先说说为什么要写这个enum枚举类吧,是群里有个新手问:怎样把enum类中的值遍历得到,其实自己用的也很少.自己也是确实不知道,于是我去网上搜了不少,总结了些,希望对大家有帮助:首先我说说怎样遍历枚 ...

  3. java.lang包—枚举类Enum

    原文作者:山高我为 原文地址:java enum的用法详解 目录 一.enum关键字 二.Enum类源码 三.疑问 四.Enum常见用法 一.enum关键字 enum关键字是在Java1.5也就是Ja ...

  4. java enum枚举类的用法以及高级玩法

    enum(枚举)类介绍 java枚举类是一组预定义常量的集合,使用enum关键字声明这个类,常量名称官方建议大写 1.enum类基本用法 举一个常见的例子,例如星期就可以描述为一个枚举类,如下 pub ...

  5. JAVA进阶教学之(Enum枚举类)

    首先,我们为什么要学习Enum枚举类 我们引入一段代码: package com.lbj.javase.enumTest;public class EnumTest01 {public static ...

  6. Enum枚举类|注解Annotation

    Enum枚举类 ①枚举类和普通类的差别: 使用 enum 定义的枚举类默认继承了 java.lang.Enum 类 枚举类的构造器仅仅能使用 private 訪问控制符 枚举类的全部实例必须在枚举类中 ...

  7. Java状态码枚举类

    前言:当浏览者访问一个网页时,浏览者的浏览器会向网页所在服务器发出请求.当浏览器接收并显示网页前,此网页所在的服务器会返回一个包含HTTP状态码的信息头(server header)用以响应浏览器的请 ...

  8. enum java 比较_Kotlin与Java比较:枚举类

    前言 Kotlin作为JVM系的语言,起源于Java又不同于Java.通过在语言层面比较两者的区别,可以使得开发者能够快速学习,融会贯通. 枚举使用场景 使用枚举的场景非常明确,即只要一个类的对象是有 ...

  9. Java中的枚举类是什么?enum关键字怎么使用?

    枚举类 文章目录 枚举类 枚举类的使用:入门 自定义枚举类 方法一:自定义枚举类 方式二: enum 关键字定义枚举类(主要用该方式) Enum类的主要方法 使用enum关键字定义的枚举类实现接口 主 ...

  10. JAVA中Switch中使用Enum枚举类

    问题 项目中出现了需要结合使用Switch和Enum的情况,按照平常思路编写代码后发现无法通过编译,如下列代码所示(只保留关键部分) 枚举类Constant代码如下 public enum Const ...

最新文章

  1. 《精通并发与Netty》学习笔记(13 - 解决TCP粘包拆包(一)概念及实例演示)
  2. Android targetSdkVersion详解
  3. 计算机视觉领域热门研究方向state-of-art算法实时更新
  4. android image 转yuv_Android将camera获取到的YuvData在jni中转化为Mat方法
  5. SIGTERM等信号含义
  6. pl/mysql安装_MySQL安装
  7. halcon学习笔记——(6)单摄像机标定
  8. linux电子教室软件,在Deepin Linux系统下安装和设置Veyon多媒体电子教室的方法
  9. split函数 在oracle,oracle的split函数
  10. php判断访问客户端是手机,判断客户端是否手机访问
  11. 山东省软件设计大赛参赛经验
  12. 史上很全的注册表修改大全
  13. spark -- PCA
  14. Excel中批量添加批注图片
  15. Vscode中搜索字符串失败,报错“Spawn ${path}\ressources\app\node_module.asar.unpacked\vs-code-ripgrep\bin\rg.exe“
  16. setfocus属性
  17. 【滤波跟踪】基于matlab捷联惯导仿真【含Matlab源码 1935期】
  18. 【深度学习】跟李沐学ai 线性回归 从零开始的代码实现超详解
  19. 14 个最佳 Vue UI 组件库推荐
  20. 手把手教程Atlas安装与使用

热门文章

  1. 步子未稳就匆忙上市,小熊电器恐作茧自缚?
  2. 简单的防止Windows自动锁屏的VBS脚本
  3. 锐龙r97900x参数 r9 7900x核显相当于什么显卡
  4. 荣耀es什么时候支持鸿蒙,荣耀手表es支持常亮吗,荣耀手表es可以常亮吗
  5. ubuntu 16.04 修改密码
  6. php去掉第一个空格,php怎么去除前面空格
  7. Oracle 查询 凌晨到现在的各个时间点
  8. Vue 创建粒子效果插件
  9. PDFkit用python批量把markdown格式文件导出成pdf文件代码
  10. 优质供应商推荐——天威诚信电子认证云服务平台