一、前言                                

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魔法堂:枚举类型详解的更多相关文章

  1. Java中的枚举类型详解

    枚举类型介绍 枚举类型(Enumerated Type) 很早就出现在编程语言中,它被用来将一组类似的值包含到一种类型当中.而这种枚举类型的名称则会被定义成独一无二的类型描述符,在这一点上和常量的定义 ...

  2. Java魔法堂:内部类详解

    一.前言 对于内部类平时编码时使用的场景不多,比较常用的地方应该就是绑定事件处理程序的时候了(从C#.JS转向Java阵营的孩子总不不习惯用匿名内部类来做事件订阅:().本文将结合Bytecode对四 ...

  3. Java中的枚举使用详解

    转载至:http://www.cnblogs.com/linjiqin/archive/2011/02/11/1951632.html package com.ljq.test; /** * 枚举用法 ...

  4. java枚举类型详解

      枚举类型是JDK1.5的新特性.显然,enum很像特殊的class,实际上enum声明定义的类型就是一个类.而这些类都是类库中Enum类的子类(java.lang.Enum<E>).它 ...

  5. MyBatis魔法堂:ResultMap详解

    一.前言   MyBatis是基于“数据库结构不可控”的思想建立的,也就是我们希望数据库遵循第三范式或BCNF,但实际事与愿违,那么结果集映射就是MyBatis为我们提供这种理想与现实间转换的手段了, ...

  6. C语言--enum,typedef enum 枚举类型详解

    原文:http://z515256164.blog.163.com/blog/static/32443029201192182854300/ 有改动 C语言详解 - 枚举类型 注:以下全部代码的执行环 ...

  7. 测者的测试技术手册:Junit单元测试遇见的一个枚举类型的坑(枚举类型详解)

    Enum的简介 枚举类型很早就在计算机语言中存在了,主要被用来将一组相似的值包含进一种类型中,这种类型的名称被定义成独一无二的类型描述符,这就是枚举类型. 在java语言中,枚举类型是一个完整功能的类 ...

  8. C++枚举类型详解

    原创作品,转载请注明来源:http://www.cnblogs.com/shrimp-can/p/5171110.html  一.枚举类型的定义 enum 类型名 {枚举值表}: 类型名是变量名,指定 ...

  9. 转载 - C - 枚举类型详解

    出处:http://www.cnblogs.com/JCSU/articles/1299051.html 注:以下全部代码的执行环境为VC++ 6.0 在程序中,可能需要为某些整数定义一个别名,我们可 ...

随机推荐

  1. 用Razor做静态页面生成器

    本来是用asp.net webpages做的博客网站,数据库用了一个陌生的本地数据库,只是觉得用起来很爽快,用新鲜的东西有一种刺激.后来数据库挂了,估计是存某个字段的时候出了问题,可是新鲜的东西,也不 ...

  2. ubuntu笔记

    安装包 sudo dpkg -i xxxx.deb 如果用户具有sudo权限,那么直接可以运行如下命令: #sudo su root #passwd #更改密码 或者直接运行sudo passwd r ...

  3. Sensor(LIGHT)

    package com.example.sensor01; import java.util.List; import android.hardware.Sensor; import android. ...

  4. tsd-提升IDE对JavaScript智能感知的能力

    在编写前端JavaScript代码时,最痛苦的莫过于代码的智能感知(Intelli Sense). 追其根源,是因为JavaScript是一门弱类型的动态语言.对于弱类型的动态语言来说,智能感知就是I ...

  5. Nim教程【十】

    openarray类型 注意:openarray类型只能用于参数 固定大小的数组虽然性能不错,但过于呆板,使用取来不是很方便 对于一个方法来说,传入参数如果是一个数组,最好是不要限制数组的长度 也就是 ...

  6. 十进制数转化成二进制后包含一的数量(c++)

    #include <iostream> using namespace std;int func(int x){    int count=0;    while(x){          ...

  7. [安卓] 12、开源一个基于SurfaceView的飞行射击类小游戏

    前言  这款安卓小游戏是基于SurfaceView的飞行射击类游戏,采用Java来写,没有采用游戏引擎,注释详细,条理比较清晰,适合初学者了解游戏状态转化自动机和一些继承与封装的技巧. 效果展示    ...

  8. 《OOC》笔记(4)——自动化地将C#代码转化为C代码(结构版)

    <OOC>笔记(4)——自动化地将C#代码转化为C代码(结构版) 我在<C表达面向对象语言的机制——C#版>中已经说明了从C#到C的转换方法.这次看<OOC>也是想 ...

  9. Atitit python3.0 3.3 3.5 3.6 新特性 Python2.7新特性1Python 3_x 新特性1python3.4新特性1python3.5新特性1值得关注的新特性1Pyth

    Atitit python3.0 3.3 3.5 3.6 新特性 Python2.7新特性1 Python 3_x 新特性1 python3.4新特性1 python3.5新特性1 值得关注的新特性1 ...

  10. Servlet过滤器,Servlet过滤器创建和配置

    第一:Servlet的过滤器的创建和配置,创建一个过滤器对象需要实现javax.servlet.Filter接口,同时实现Filter的3个方法.        第一方法是过滤器中的init()方法用 ...