在JDK1.5 之前,我们定义常量都是: public static fianl.... 。现在好了,有了枚举,可以把相关的常量分组到一个枚举类型里,而且枚举提供了比常量更多的方法。
一. 什么是枚举
枚举是一种数据类型,具有集合的一些特点,可以存放多个元素,但存储对象有限且固定,枚举也有比较常见的使用场景,如我们需要表达性别(男、女),颜色(红、黄、蓝),星期(星期一、星期二...星期日),四季(春、夏、秋、冬),地理位置(东、西、南、北),方向(前、后、左、右)等,这些场景都非常适合枚举。
二. 定义枚举
java中使用enum
来定义枚举,和class
,interface
具有同样的等级,(注意是enum,而不是Enum),定义枚举可以有两种情况
第一种:默认构造器(空构造器)
public enum Quarter {
SPRING, SUMMER, AUTUMN, WINTER;
}
未定义成员变量和成员方法,省略了 private Quarter() {}
。
public enum Quarter {
SPRING, SUMMER, AUTUMN, WINTER;
private Quarter(){}
}
第二种:定义了成员变量、带参构造器
枚举可以定义成员变量,包括成员属性、方法,抽象方法,静态方法等
public enum Quarter {
SPRING("春"), SUMMER("夏"), AUTUMN("秋"), WINTER("冬");
private Quarter(String name){
this.name = name;
}
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
//静态方法
public static void printName(){
System.out.println(Quarter.SUMMER);
}
//抽象方法
public abstract void printValue();
}
注意:枚举和类一样,可以有多个构造器,即有了一个带参构造器,还可以有无参构造器,编译是可以通过的。
除此之外:
- 1.枚举enum 默认继承了java.lang.Enum类,实现了 java.lang.Seriablizable 和 java.lang.Comparable 两个接口,可序列化以及进行比较;
- 2.所有的枚举值都是常量,默认采用了public static final 进行了修饰,enum不是类,自然也不能被继承或实现;
- 3.枚举值必须在第一行,否则编译出错;
三. 枚举常见用法
1.获取枚举元素以及成员变量
public static void main(String[] args) {
//获取枚举元素
System.out.println(Quarter.SPRING);
//调用toString()方法将枚举元素转化为String类型
System.out.println(Quarter.SPRING.name());
System.out.println(Quarter.SPRING.toString());
//获取成员变量
System.out.println(Quarter.SPRING.getName());
}
输出结果:
SPRING
SPRING
SPRING
春
2.枚举遍历
有时候我们需要将枚举元素都取出来作为查询条件,此时就需要进行遍历,通过调用Quarter.values()方法
public static void main(String[] args) {
//通过values()方法得到Quarter元素的数组
Quarter[] quarters = Quarter.values();
List<String> quarterParam = new ArrayList<>(quarters.length);
//将数组中的元素存储到List集合quarterParam中
for (Quarter quarter : quarters) {
quarterParam.add(quarter.toString());
}
}
3.switch条件判断
枚举的元素有限且固定,通过switch做条件判断却是正好
public static void main(String[] args){
Quarter quarter = Quarter.AUTUMN;
switch (quarter) {
case SPRING:
System.out.println("找到了,是它:" + SPRING);
break;
case SUMMER:
System.out.println("找到了,是它:" + SUMMER);
break;
case AUTUMN:
System.out.println("找到了,是它:" + AUTUMN);
break;
case WINTER:
System.out.println("找到了,是它:" + WINTER);
break;
default:
System.out.println("未找到");
break;
}
}
执行后结果:
找到了,是它:AUTUMN
简单说明一下:其实此处不需要default的内容,或者说将WINTER作为default即可,因为枚举天然有类型限制,你只能传递它已有的元素以及null,不过正常情况下都会提前判断传递的参数是否为null,当switch接收到null时会报java.lang.NullPointerException。
4.枚举比较
4.1. 判断是否相等
枚举判断两个元素是否相等直接使用==进行判断即可,因为它非类,也无法实例化,存储位置自然也不会根据对象的不同而不同。
public static void main(String[] args){
System.out.println(Quarter.AUTUMN == Quarter.AUTUMN);
System.out.println(Quarter.AUTUMN == Quarter.WINTER);
}
执行结果:
true
false
也可以通过equals()
方法进行比较,不过没必要,因为其底层也是通过 ==
来实现的。
public abstract class Enum<E extends Enum<E>>
implements Comparable<E>, Serializable {
//equals比较
public final boolean equals(Object other) {
return this==other;
}
}
4.2. 顺序比较
通过compareTo进行枚举元素比较,此处比较的是在枚举中元素的先后顺序,返回的是位置序数的差值
public static void main(String[] args){
System.out.println(Quarter.AUTUMN.compareTo(Quarter.SPRING));
System.out.println(Quarter.AUTUMN.compareTo(Quarter.WINTER));
}
执行结果为:
2
-1
我们看一下compareTo方法
public abstract class Enum<E extends Enum<E>>
implements Comparable<E>, Serializable {
//元素在枚举中的序号
private final int ordinal;
//可以看到compareTo方法返回的就是两个元素ordinal的差值
public final int compareTo(E o) {
Enum<?> other = (Enum<?>)o;
Enum<E> self = this;
if (self.getClass() != other.getClass() &&
self.getDeclaringClass() != other.getDeclaringClass())
throw new ClassCastException();
return self.ordinal - other.ordinal;
}
}
5.枚举也可也实现接口
public interface WeatherInterface {
//获取温度
public String getTemperature(Quarter quarter);
}
public enum Quarter implements WeatherInterface{
SPRING("春"), SUMMER("夏"), AUTUMN("秋"), WINTER("冬");
private Quarter(String name){
this.name = name;
}
private String name;
public String getName() {
return name;
}
//重写获取温度方法
@Override
public String getTemperature(Quarter quarter) {
switch (quarter) {
case SPRING:
return "适中";
case SUMMER:
return "热";
case AUTUMN:
return "凉爽";
case WINTER:
return "寒冷";
default:
return "无法确定";
}
}
}
但注意枚举不可继承其他类
public class Pquarter {}
//无法编译通过
public enum Quarter extends Pquarter {}
6.使用接口来组织枚举
如果枚举太多也不好管理,同时条理不够清晰,我们可以通过接口来将多个相关枚举组织在一起进行管理,
public interface Weather {
enum Quarter implements Weather {
SPRING, SUMMER, AUTUMN, WINTER;
}
enum Temperature implements Weather {
MODERATE, HEAT, COOL, COLD
}
//调用时直接通过Weather.Quarter.SPRING即可。
}
四.实战
Quarter.java
单个参数枚举类
package cn.how2j;
/**
* Created by kaiyiwang on 20/7/7.
*/
public enum Quarter {
SPRING("春"), SUMMER("夏"), AUTUMN("秋"), WINTER("冬");
private Quarter(String name){
this.name = name;
}
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
//静态方法
public static void printName(){
System.out.println(Quarter.SUMMER);
}
}
两个参数枚举类:QuarterTwo.java
package cn.how2j;
/**
* Created by kaiyiwang on 20/7/7.
*/
public enum QuarterTwo {
SPRING("春", 1), SUMMER("夏", 2), AUTUMN("秋", 3), WINTER("冬", 4);
private QuarterTwo(String name, int value){
this.name = name;
this.value = value;
}
private String name;
private int value;
public String getName() {
return name;
}
public int getValue() {
return value;
}
public void setName(String name) {
this.name = name;
}
public void setValue(int value) {
this.value = value;
}
//静态方法
public static void printName(){
System.out.println(QuarterTwo.SUMMER);
}
}
测试类 TestEnum.java
package cn.how2j;
/**
* Created by kaiyiwang on 20/7/7.
*/
public class TestEnum {
public static void main(String[] args) {
// 获取枚举元素
System.out.println(Quarter.SPRING);
//调用toString()方法将枚举元素转化为String类型
System.out.println(Quarter.SPRING.name());
System.out.println(Quarter.SPRING.toString());
//获取成员变量
System.out.println(Quarter.SPRING.getName());
// 多参数构建
System.out.println("-------- TWO --------");
System.out.println(QuarterTwo.SPRING);
System.out.println(QuarterTwo.SPRING.name());
System.out.println(QuarterTwo.SPRING.getName());
System.out.println(QuarterTwo.SPRING.getValue());
}
}
打印结果:
/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/bin/java -Didea.launcher.port=7534 "-Didea.launcher.bin.path=/Applications/IntelliJ IDEA 15.app/Contents/bin" -Dfile.encoding=UTF-8 -classpath "/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/charsets.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/deploy.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/ext/cldrdata.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/ext/dnsns.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/ext/jaccess.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/ext/jfxrt.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/ext/localedata.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/ext/nashorn.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/ext/sunec.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/ext/sunjce_provider.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/ext/sunpkcs11.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/ext/zipfs.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/javaws.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/jce.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/jfr.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/jfxswt.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/jsse.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/management-agent.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/plugin.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/resources.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/jre/lib/rt.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/lib/ant-javafx.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/lib/dt.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/lib/javafx-mx.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/lib/jconsole.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/lib/packager.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/lib/sa-jdi.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_181.jdk/Contents/Home/lib/tools.jar:/Users/kaiyiwang/Code/java/rabbitmq_fanout/target/classes:/WEB/java/maven/repository/com/rabbitmq/amqp-client/3.6.5/amqp-client-3.6.5.jar:/WEB/java/maven/repository/cn/hutool/hutool-all/4.3.1/hutool-all-4.3.1.jar:/Applications/IntelliJ IDEA 15.app/Contents/lib/idea_rt.jar" com.intellij.rt.execution.application.AppMain cn.how2j.TestEnum
SPRING
SPRING
SPRING
春
-------- TWO --------
SPRING
SPRING
春
1
Process finished with exit code 0
其他:
enum MallSalesOrderStatus{
UNKNOW("未知", 0),
WAIT_BUYER_PAY("等待买家支付", 10),
WAIT_BUYER_CONFIRM_GOODS("等待买家确认收货", 30),
REFUND_FINISHED("退款完成", 60);
private final String caption;
private final int value;
MallSalesOrderStatus(String caption, int value){
this.caption = caption;
this.value = value;
}
}
public class Test{
public static void main(String[] args)
{
MallSalesOrderStatus status = MallSalesOrderStatus.WAIT_BUYER_PAY;
System.out.print(status);
}
}
五. 总结
本文主要介绍了枚举enum主要特点,定义,无参构造器,有参构造器,枚举的常用方法举例以及主要的应用场景。
转载文章:
Java的枚举enum示例详解
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。