Java魔法堂:枚举类型详解
一、前言
Java的枚举类型相对C#来说具有更灵活可配置性,Java的枚举类型可以携带更多的信息。
// C#
enum MyColor{
RED = ,
BLUE =
}
Console.Write(MyColor.RED); // Java
enum MyColor{
RED("Hot", ), BLUE("SAD",); private String mood;
public String getMood{
return mood;
}
private int index;
public int getIndex(){
return index;
}
private MyColor(String mood, int index){
this.mood = mood;
this.index = index;
}
}
System.out.println(MyColor.RED.getMood());
本文将对枚举类型进行较为详细的叙述,以便日后查阅。
二、最简单的用法——常量
/* 定义 */
// 形式1
enum MyColor{
RED,BLUE
} // 形式2
enum MyColor{
RED,BLUE;
} /* 使用 */
System.out.println(MyColor.RED.name()); // 显示RED
System.out.println(MyColor.RED.ordinal()); // 显示0
System.out.println(MyColor.BLUE.name()); // 显示BLUE
System.out.println(MyColor.BLUE.ordinal()); // 显示1
枚举值的name()会返回枚举值的字面量,而ordinal()为返回枚举值的索引,而索引是以枚举值定义时的位置来确定,并在编译时设置的。下面我们来看看到底编译器为我们做了什么?
final class MyColor extends java.lang.Enum<MyCorlor>{
public static final MyColor RED;
flags: ACC_PUBLIC, ACC_STATIC, ACC_FINAL, ACC_ENUM
public static final MyColor BLUE;
flags: ACC_PUBLIC, ACC_STATIC, ACC_FINAL, ACC_ENUM
public static MyColor[] values();
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=, locals=, args_size=
: getstatic # // Field $VALUES:[LMyColor;
: invokevirtual # // Method "[LMyColor;".clone:()Ljava/lang/Object;
: checkcast # // class "[LMyColor;"
: areturn
LineNumberTable:
line :
public static MyClass valueOf(java.lang.String);
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=, locals=, args_size=
: ldc_w # // class MyColor
: aload_0
: invokestatic # // Method java/lang/Enum.valueOf:(Ljava/lang/Class;Ljava/lang/String;)Ljava/lang/Enum;
: checkcast # // class MyColor
: areturn
LineNumberTable:
line :
static {};
flags: ACC_STATIC
Code:
stack=, locals=, args_size=
: new # // class MyColor
: dup
: ldc # // String RED
: iconst_0
: invokespecial # // Method "<init>":(Ljava/lang/String;I)V
: putstatic # // Field RED:LMyColor;
: new # // class MyColor
: dup
: ldc # // String BLUE
: iconst_1
: invokespecial # // Method "<init>":(Ljava/lang/String;I)V
: putstatic # // Field BLUE:LMyColor;
: iconst_2
: anewarray # // class MyColor
: dup
: iconst_0
: getstatic # // Field RED:LMyColor;
: aastore
: dup
: iconst_1
: getstatic # // Field BLUE:LMyColor;
: aastore
: putstatic # // Field $VALUES:[LMyColor;
: return
LineNumberTable:
line :
line : 26
}
可以看到编译器将enum MyColor编译为一个继承Enum<MyColor>并且带修饰符final的MyColor类。
而枚举值RED和BLUE则被编译为MyColor的类常量,并且在类加载的初始化阶段实例化。MyColor默认的构造函数会调用父类Enum<MyColor>的构造函数Enum<E>(String name, int ordinal)来设置私有字段name和ordinal的值。其中iconst_0和iconst_1分别表示将0和1压栈,invokespecial #8则是调用构造函数Enum<E>(String name, int ordinal)。
: new # // class MyColor
: dup
: ldc # // String RED
: iconst_0 // int 0
: invokespecial # // Method "<init>":(Ljava/lang/String;I)V
: putstatic # // Field RED:LMyColor;
: new # // class MyColor
: dup
: ldc # // String BLUE
: iconst_1 // int 1
: invokespecial # // Method "<init>":(Ljava/lang/String;I)V
: putstatic # // Field BLUE:LMyColor;
另外在类加载的初始化阶段会生成一个私有的$VALUE数组用于存放常量RED和BLUE,而在调用MyColor.values()返回的正是这个$VALUE数组的复制品。
: iconst_2
: anewarray # // class MyColor
: dup
: iconst_0
: getstatic # // Field RED:LMyColor;
: aastore
: dup
: iconst_1
: getstatic # // Field BLUE:LMyColor;
: aastore
: putstatic # // Field $VALUES:[LMyColor;
小结:
1. 定义枚举类型本质上就是在定义带final修饰符的Enum<E>的子类;
2. 枚举值本质为第1点所定义的类的类常量;
3. 枚举值的ordinal值由其定义时的排序决定,并且在编译时已经被设置好了。
三、枚举类型的抽象父类Enum<E>
其实我们大多数情况下都是调用父类Enum<E>的方法来操作自定义的枚举值,下面一起看看父类Enum<E>吧!
1. 它为抽象类且继承了Comparable<E>和Serializable两个类。
2. 内含私有字段name和ordinal和对应的公有get方法name()和ordinal()。
3. 重写了equals方法,通过==比较两个枚举值的内存地址来判断两者是否相同。
4. 实现compareTo方法,通过比较两个枚举值的ordinal值来做判断。
5. getDeclaringClass方法,用于返回枚举的Class对象。
四、携带更多信息——自定义构造函数
由于枚举最终被编译为类,因此我们通过自定义构造函数、自定义字段和方法来让枚举值携带更多信息
public enum MyColor{
RED("Hot", ), BLUE("SAD",);
private String mood;
private int index;
private MyColor(String mood, int index){
this.mood = mood;
this.index = index;
}
}
注意:
1. 自定义的构造函数必须是私有的;
2. 构造函数内不能显式调用父类的构造函数;
3. RED、BLUE的ordinal值依然是0和1,那么值依然是RED和BLUE。
上述3点规定和结果的原因是编译器会对我们的自定义构造函数进行加工变为
private MyColor(String name, String ordinal, String mood, int index){
super(name, ordinal);
this.mood = mood;
this.index = index;
}
五、让相同枚举类型下的枚举值具有不同的行为——重写枚举值的方法
public enum MyColor{
RED, BLUE(){
@Override
public boolean getFlag(){
return false;
}
};
public boolean getFlag(){
return true;
}
}
// 调用
System.out.println(MyColor.RED.getFlag()); // 显示true
System.out.println(MyColor.BLUE.getFlag()); // 显示false
可以看到枚举值RED和BLUE同一个方法具有不同的行为。其实这是通过匿名内部类的方式实现的,BLUE的类型为MyColor$1 extends MyColor,而RED的类型为MyColor。
六、使用接口组织枚举
public interface Food {
enum Coffee implements Food {
BLACK_COFFEE, DECAF_COFFEE, LATTE, CAPPUCCINO
}
enum Dessert implements Food {
FRUIT, CAKE, GELATO
}
}
七、总结
若有纰漏请大家指正,谢谢。
尊重原创,转载请注明来自:http://www.cnblogs.com/fsjohnhuang/p/4297741.html ^_^肥仔John
八、参考
http://www.tuicool.com/articles/YvQZFf
http://www.cnblogs.com/hemingwang0902/archive/2011/12/29/2306263.html
http://www.cnblogs.com/frankliiu-java/archive/2010/12/07/1898721.html
Java魔法堂:枚举类型详解的更多相关文章
- Java中的枚举类型详解
枚举类型介绍 枚举类型(Enumerated Type) 很早就出现在编程语言中,它被用来将一组类似的值包含到一种类型当中.而这种枚举类型的名称则会被定义成独一无二的类型描述符,在这一点上和常量的定义 ...
- Java魔法堂:内部类详解
一.前言 对于内部类平时编码时使用的场景不多,比较常用的地方应该就是绑定事件处理程序的时候了(从C#.JS转向Java阵营的孩子总不不习惯用匿名内部类来做事件订阅:().本文将结合Bytecode对四 ...
- Java中的枚举使用详解
转载至:http://www.cnblogs.com/linjiqin/archive/2011/02/11/1951632.html package com.ljq.test; /** * 枚举用法 ...
- java枚举类型详解
枚举类型是JDK1.5的新特性.显然,enum很像特殊的class,实际上enum声明定义的类型就是一个类.而这些类都是类库中Enum类的子类(java.lang.Enum<E>).它 ...
- MyBatis魔法堂:ResultMap详解
一.前言 MyBatis是基于“数据库结构不可控”的思想建立的,也就是我们希望数据库遵循第三范式或BCNF,但实际事与愿违,那么结果集映射就是MyBatis为我们提供这种理想与现实间转换的手段了, ...
- C语言--enum,typedef enum 枚举类型详解
原文:http://z515256164.blog.163.com/blog/static/32443029201192182854300/ 有改动 C语言详解 - 枚举类型 注:以下全部代码的执行环 ...
- 测者的测试技术手册:Junit单元测试遇见的一个枚举类型的坑(枚举类型详解)
Enum的简介 枚举类型很早就在计算机语言中存在了,主要被用来将一组相似的值包含进一种类型中,这种类型的名称被定义成独一无二的类型描述符,这就是枚举类型. 在java语言中,枚举类型是一个完整功能的类 ...
- C++枚举类型详解
原创作品,转载请注明来源:http://www.cnblogs.com/shrimp-can/p/5171110.html 一.枚举类型的定义 enum 类型名 {枚举值表}: 类型名是变量名,指定 ...
- 转载 - C - 枚举类型详解
出处:http://www.cnblogs.com/JCSU/articles/1299051.html 注:以下全部代码的执行环境为VC++ 6.0 在程序中,可能需要为某些整数定义一个别名,我们可 ...
随机推荐
- pdf.js在IIS中配置使用笔记
最近在手机App开发Android版本时候遇到需要显示PDF文件的需求,记得之前直接使用系统浏览器或者WebView就可以显示,但是现在不可以了,只能另寻其他办法. 最终找到PDF.JS来进行实现,但 ...
- Oracle Enterprise Manager打不开的解决方法
之前OEM一直可以打开,但今天上班发现打不开了,输入http://localhost:1158/em 提示该网页无法打开. 那么检查一下: cmd进命令行 C:\Documents and Setti ...
- 算法:POJ1008 Maya Calendar
此题非常水,不做说明. package practice; import java.io.BufferedInputStream; import java.util.Scanner; /** * @a ...
- Java设计模式6:策略模式
策略模式 策略模式的用意是针对一组算法,将每一个算法封装到具有共同接口的独立类中,从而使得它们可以相互替换.策略模式使得算法可以在不影响到客户端的情况下发生变化. 策略模式的结构 策略模式是对算法的包 ...
- 5天玩转C#并行和多线程编程 —— 第一天 认识Parallel
5天玩转C#并行和多线程编程系列文章目录 5天玩转C#并行和多线程编程 —— 第一天 认识Parallel 5天玩转C#并行和多线程编程 —— 第二天 并行集合和PLinq 5天玩转C#并行和多线程编 ...
- Oracle存在修改,不存在插入记录
接触编程以来,在数据存储方面一直用的MS SQL.Oracle这名字对我来说是如此的熟悉,但是对其内容却很陌生,最近公司的一个项目用起了Oracle,所以也开始高调的用起了Oracle.在没有接触Or ...
- 自动备份文件到GITHUB的方法
由于一个制作着玩的项目需要制作上传文件的功能,自己又不是搞网站的,也不想去维护一个服务器. 于是开发了一个上传服务器,可以自动把我上传到服务器的数据同步到Github服务器 而github服务器又提供 ...
- Unity3D使用经验总结 编辑器扩展篇
一个引擎,最重要的就是工具,工具除了提升开发速度,提供可视化操作环境以外,还带了容错功能. 它使得大家的工作局限在一定的范围内,比如一个变量的配置,或者是一些类型的选择. 使用编辑器,使得既使不太明白 ...
- ios 常用数学函数
需要 引入头文件 #import <math.h> 1. 三角函数 double sin (double);正弦 double cos (double);余弦 double tan ...
- EF架构~引入规约(Specification)模式,让程序扩展性更强
回到目录 规约(Specification)模式:第一次看到这东西是在microsoft NLayer项目中,它是微软对DDD的解说,就像petshop告诉了我们MVC如何使用一样,这个规约模式最重要 ...