JAVA中枚举Enum详解
1.关键字:enum。枚举可以定义成单独的文件,也可以定义在其他类内部。
枚举在类内部的示例:
public class EnumInner {
public static void main(String[] args) {
Day day=Day.MONDAY;
}
enum Day{
MONDAY,TUESDAY,THIRSDAY,FORTHDAY,FRIDAY
}
}
Enum里,需要我们定义实例,定义的实例,返回类型还是Enum这个类本身。实例之间,用逗号隔开。我们也可以给实例添加参数,如MONDAY("参数1","参数2","参数3",,,)。添加参数后,我们在后面,定义成员变量,来接收这些参数。因为MONDAY,TUESDAY这些,都是Enum实例,其定义的参数,就是成员变量,只不过,在Eunm里,实例和成员变量,混在一起来写了。然后,我们提供方法,来获得这些实例的参数。如下代码:
public enum SimpleEnum {
MONDAY("aaa","AAA"),//一个实例,相当于是一个对象。aaa就是成员变量的值
TUESDAY("bbb");//另一个实例
//实例中写了几个参数,就写几个成员变量
private String param1;//第一个成员变量
private String param2;//第二个成员变量
/**
* 枚举可以定义构造方法,但是构造方法都是私有的,或者是默认修饰符,不能用public修饰符,编译都过不去
* 因为构造方法我们不能手动调用,是编译器自己调用的
* 在实例中,我们有几个参数,定义构造器时,就要写几个参数的构造方法,这是为了给成员变量赋值。
* @param param1
* @param param2
*/
private SimpleEnum(String param1,String param2) {
this.param1=param1;
this.param2=param2;
}
private SimpleEnum(String param){
this.param1=param;
}
/**
* 通过提供方法,供外部获得实例的参数值。在调用这些方法时,都是通过枚举.实例.方法调用。直接使用枚举,不.实例,是不能用这些方法的
* @return
*/
public String getParam(){
return param1;
}
public String getParam2(){
return param2;
}
public static void main(String[] args) {
SimpleEnum.MONDAY.getParam();//通过枚举.实例.方法,获得参数
}
}
2.枚举原理解析
我们定义的所有的枚举,在jdk内部,都会自动继承java.lang.Enum类。这是一个抽象类,我们看源码的构造方法:
//由编译器调用
protected Enum(String name, int ordinal) {
this.name = name;
this.ordinal = ordinal;
}
枚举的构造方法都是由编译器自己调用的,我们在定义枚举时,无需定义构造方法。
我们说到,枚举在编译器中会自动继承Enum类,因为java是单继承的,所以,我们定义的枚举,是不能手动继承任何类的。但是,在枚举中,可以定义成员变量和方法,甚至可以定义main方法。也可以定义抽象方法,然后在枚举实例中,实现这些抽象方法,如下代码:
public enum SimpleEnum {
MONDAY("aaa","AAA") {
//实例中实现抽象方法
@Override
public void enumAbatract() {
// TODO Auto-generated method stub
System.out.println("星期一");
}
},//一个实例,相当于是一个对象。aaa就是成员变量的值
TUESDAY("bbb") {
@Override
public void enumAbatract() {
// TODO Auto-generated method stub
System.out.println("星期二");
}
};//另一个实例
//实例中写了几个参数,就写几个成员变量
private String param1;//第一个成员变量
private String param2;//第二个成员变量
/**
* 枚举可以定义构造方法,但是构造方法都是私有的,或者是默认修饰符,不能用public修饰符,编译都过不去
* 因为构造方法我们不能手动调用,是编译器自己调用的
* 在实例中,我们有几个参数,定义构造器时,就要写几个参数的构造方法,这是为了给成员变量赋值。
* @param param1
* @param param2
*/
private SimpleEnum(String param1,String param2) {
this.param1=param1;
this.param2=param2;
}
private SimpleEnum(String param){
this.param1=param;
}
/**
* 通过提供方法,供外部获得实例的参数值。在调用这些方法时,都是通过枚举.实例.方法调用。直接使用枚举,不.实例,是不能用这些方法的
* @return
*/
public String getParam(){
return param1;
}
public String getParam2(){
return param2;
}
//定义抽象方法,上面每个实例,需要实现这个方法。
public abstract void enumAbatract();
public static void main(String[] args) {
SimpleEnum.MONDAY.getParam();
SimpleEnum.MONDAY.enumAbatract();
}
}
由此可知,enum就是一个java类,类里面的东西,在枚举里都可以有。定义了抽象方法,每个实例实现不同的抽象方法,我们获得每个实例时,也就有了不同的实现方法。
同样的,枚举也能实现接口,然后实现接口里的方法,代码如下:
/**
* 枚举接口,随便定义两个方法
* @author Administrator
*
*/
public interface IEnum { public void method(); public void way(); }
/**
* 枚举实现接口
* @author Administrator
*
*/
public enum EnumImpl implements IEnum{ INSTANCE{
//在实例里,也可以重写实现的接口方法
@Override
public void method() {
// TODO Auto-generated method stub
super.method();
}
};
/**
* 枚举中实现接口方法,那么所有的实例都是走的这个方法
*/
@Override
public void method() {
System.out.println("实现了method方法"); } @Override
public void way() {
System.out.println("实现了way方法"); } }
3.枚举与单例
在《设计模式之单例模式(Singleton Pattern)》文章中提到,枚举实现单例是最安全的,解决了反射和序列化破坏单例的可能。为什么呢?首先,我们看反射处理枚举时是怎么处理的:

这是jdk中new Instance方法的源码,我们可以看出,当通过反射实例化一个枚举时,会抛出异常。所有,我们无法通过反射,来破坏单例模式。
在序列化问题上,jvm对枚举的序列化做了特殊的规定,其序列化时,只是序列化了枚举的name()方法,在反序列化时,也保证了枚举的唯一性。相当于把枚举的唯一性,交给了jvm做保证。具体的原理,可以看这篇文章:https://blog.csdn.net/wang__qin/article/details/79802628
在这里,我们不对其进行过深的研究。
4.总结:
枚举最常用的就是取代我们之前的public static final常量定义,而是通过枚举定义一些常量参数。在枚举中,实例定义在枚举类中,成员属性和成员方法定义在枚举类中,这是一个比较特殊的地方。它不像普通的类,实例是通过new出来的,而是直接就在枚举类中定义实例,然后初始化实例的成员变量,方法等。
JAVA中枚举Enum详解的更多相关文章
- Java 枚举(enum) 详解7种常见的用法
Java 枚举(enum) 详解7种常见的用法 来源 https://blog.csdn.net/qq_27093465/article/details/52180865 JDK1.5引入了新的类型— ...
- java中的注解详解和自定义注解
一.java中的注解详解 1.什么是注解 用一个词就可以描述注解,那就是元数据,即一种描述数据的数据.所以,可以说注解就是源代码的元数据.比如,下面这段代码: @Override public Str ...
- Java中dimension类详解
Java中dimension类详解 https://blog.csdn.net/hrw1234567890/article/details/81217788
- [转载]java中import作用详解
[转载]java中import作用详解 来源: https://blog.csdn.net/qq_25665807/article/details/74747868 这篇博客讲的真的很清楚,这个作者很 ...
- Java中日志组件详解
avalon-logkit Java中日志组件详解 lanhy 发布于 2020-9-1 11:35 224浏览 0收藏 作为开发人员,我相信您对日志记录工具并不陌生. Java还具有功能强大且功能强 ...
- Java 枚举(enum) 详解4种常见的用法
JDK1.5引入了新的类型——枚举.在 Java 中它虽然算个“小”功能,却给我的开发带来了“大”方便. 大师兄我又加上自己的理解,来帮助各位理解一下. 用法一:常量 在JDK1.5 之前,我们定义常 ...
- Java 枚举 enum 详解
本文部分摘自 On Java 8 枚举类型 Java5 中添加了一个 enum 关键字,通过 enum 关键字,我们可以将一组拥有具名的值的有限集合创建为一种新的类型,这些具名的值可以作为常规的程序组 ...
- 关于Java中枚举Enum的深入剖析
在编程语言中我们,都会接触到枚举类型,通常我们进行有穷的列举来实现一些限定.Java也不例外.Java中的枚举类型为Enum,本文将对枚举进行一些比较深入的剖析. 什么是Enum Enum是自Java ...
- Java中反射机制详解
序言 在学习java基础时,由于学的不扎实,讲的实用性不强,就觉得没用,很多重要的知识就那样一笔带过了,像这个马上要讲的反射机制一样,当时学的时候就忽略了,到后来学习的知识中,很多东西动不动就用反射, ...
随机推荐
- POJ - 1654 利用叉积求三角形面积 去 间接求多边形面积
题意:在一个平面直角坐标系,一个点总是从原点出发,但是每次移动只能移动8个方向的中的一个并且每次移动距离只有1和√2这两种情况,最后一定会回到原点(以字母5结束),请你计算这个点所画出图形的面积 题解 ...
- hdu5432 Pyramid Split
Problem Description Xiao Ming is a citizen who's good at playing,he has lot's of gold cones which ha ...
- Codeforces Round #672 (Div. 2) D. Rescue Nibel! (思维,组合数)
题意:给你\(n\)个区间,从这\(n\)区间中选\(k\)个区间出来,要求这\(k\)个区间都要相交.问共有多少种情况. 题解:如果\(k\)个区间都要相交,最左边的区间和最右边的区间必须要相交,即 ...
- .Net Core 国际化
创建项目什么的就不说了吧 直接进入正题吧 我这里建的是个webapi 添加资源文件 1.首先我们创建一个Language文件夹,这就是我们在后面Startup类中需要配置的目录名. 2.然后我们在La ...
- CF1471-B. Strange List
CF1471-B. Strange List 题意: 给定一个由\(n\)个数字组成的数组以及一个\(x\).现在从前往后遍历数组,若当前遍历的数字\(a[i]\)可以被\(x\)整除,那么就在数组的 ...
- 【转】Dockerfile
1. 关于docker build docker build可以基于Dockerfile和context打包出一个镜像,其中context是一系列在PATH或URL中指定的位置中的文件(contex ...
- oslab oranges 一个操作系统的实现 实验四 认识保护模式(三):中断异常
实验目的: 理解中断与异常机制的实现机理 对应章节:第三章3.4节,3.5节 实验内容: 1. 理解中断与异常的机制 2. 调试8259A的编程基本例程 3. 调试时钟中断例程 4. 建立IDT,实现 ...
- codefroces 7C
C. Line time limit per test 1 second memory limit per test 256 megabytes input standard input output ...
- Google PageSpeed Insights : 网站性能优化检测工具
1 1 https://developers.google.com/speed/pagespeed/insights/ PageSpeed Insights 使您的网页在所有设备上都能快速加载. 分析 ...
- 小程序 & taro 踩坑指南
小程序 & taro 踩坑指南 微信开发者工具, 不支持 react bug https://github.com/NervJS/taro/issues/5042 solution just ...