java枚举与.net中的枚举区别
通过一段时间的项目实践,发现java中的枚举与.net中的枚举有很大的差别,初期造成了我对java中的枚举一些错误理解及部分有缺陷的应用,其实追其原因还是因为我会习惯性的认为java的枚举在作用以及定义上与.net应该是差不多的,毕竟两者都是高级语言,语言上也有很多相似之处。这就是老师傅常说的新手好教,老兵不好教的原因,新手脑子一片空白不会有任何干扰,老兵总会以自己曾经的某些经验与新知识做对比。

  
  习惯性观点一:枚举的定义应该与.net相同,比如在.net中我们可以这样定义枚举。
public enum EItemDataType
{
Real=,
Service=
}
但java中并不能如此潇洒的书写枚举,可能需要类似这样写:
public enum EItemDataType {
    Real(1),Service(2);
    private int value;
    private EItemDataType(int value) {
        this.value = value;
    }
    public int getValue() {
        return value;
    }
    public static EItemDataType valueOf(int value) {
        switch (value) {
        case 1:
            return EItemDataType.Real;
        case 2:
            return EItemDataType.Service;
        default:
            return null;
        }
    }
}
发现.net要比java简单的多,注意几个方法:
- valueOf的方法:看作用是为了根据一个枚举的数值来得到枚举,这个功能很常见,但在.net中就不需要这样麻烦了,可以直接将数据强转成枚举,比如:
 
var itemType=(EItemDataType);
- getValue的方式,明显是需要将一个枚举转换成它所对应的值,.net中也不需要调用方法来取值,也可以强转,比如:
 
var itemTypeValue=(int)EItemDataType.Real;
- 私有构造函数,我们可以传多少参数,比如常见的我们需要显示这个枚举值对应的中文描述,在java中我们只需要在构造函数中增加一个name参数就可以了,但在.net中因为没有这货不能这样做,但可以通过 Atrribute来完成。
 
public enum EItemDataType
{
[Description("实物")]
Real=,
[Description("服务")]
Service=
}
习惯性观点二:因为.net的枚举是个值类型,所以我理所当然的会认为java的枚举也是一个值类型。之前对.net的理解就是将一些数值以更加可读性的方式体现在程序中,比如订单状态,订单类型等等,比如:
//枚举值可读性更强
if(orderInfo.orderStatus.equals(EOrderStatus.Shipped)){
//do something
} //一般不这样写,0可读性不强
if(orderInfo.orderStatus==0){
//do something
}
枚举类型的自说明:
- 编译后的文件中找到了EItemDataType.class这个文件,这说明java的枚举其实和普通的类是一样的,既然是一个类,那么肯定不是值类型了,下图中的引用类型中包含class type。
 

编译之后所对应的字节码到底是什么样的:
public final class EItemDataType extends java.lang.Enum<EItemDataType> {
  public static final EItemDataType Real;
  public static final EItemDataType Service;
  static {};
    Code:
       0: new           #1                  // class EItemDataType
       3: dup
       4: ldc           #15                 // String Real
       6: iconst_0
       7: iconst_1
       8: invokespecial #16                 // Method "<init>":(Ljava/lang/String;II)V
      11: putstatic     #20                 // Field Real:LEItemDataType;
      14: new           #1                  // class EItemDataType
      17: dup
      18: ldc           #22                 // String Service
      20: iconst_1
      21: iconst_2
      22: invokespecial #16                 // Method "<init>":(Ljava/lang/String;II)V
      25: putstatic     #23                 // Field Service:LEItemDataType;
      28: iconst_2
      29: anewarray     #1                  // class EItemDataType
      32: dup
      33: iconst_0
      34: getstatic     #20                 // Field Real:LEItemDataType;
      37: aastore
      38: dup
      39: iconst_1
      40: getstatic     #23                 // Field Service:LEItemDataType;
      43: aastore
      44: putstatic     #25                 // Field ENUM$VALUES:[LEItemDataType;
      47: return
  public int getValue();
    Code:
       0: aload_0
       1: getfield      #32                 // Field value:I
       4: ireturn
  public static EItemDataType valueOf(int);
    Code:
       0: iload_0
       1: tableswitch   { // 1 to 2
                     1: 24
                     2: 28
               default: 32
          }
      24: getstatic     #20                 // Field Real:LEItemDataType;
      27: areturn
      28: getstatic     #23                 // Field Service:LEItemDataType;
      31: areturn
      32: aconst_null
      33: areturn
  public static EItemDataType[] values();
    Code:
       0: getstatic     #25                 // Field ENUM$VALUES:[LEItemDataType;
       3: dup
       4: astore_0
       5: iconst_0
       6: aload_0
       7: arraylength
       8: dup
       9: istore_1
      10: anewarray     #1                  // class EItemDataType
      13: dup
      14: astore_2
      15: iconst_0
      16: iload_1
      17: invokestatic  #42                 // Method java/lang/System.arraycopy:(Ljava/lang/Object;ILjava/lang/Object;II)V
      20: aload_2
      21: areturn
  public static EItemDataType valueOf(java.lang.String);
    Code:
       0: ldc           #1                  // class EItemDataType
       2: aload_0
       3: invokestatic  #49                 // Method java/lang/Enum.valueOf:(Ljava/lang/Class;Ljava/lang/String;)Ljava/lang/Enum;
       6: checkcast     #1                  // class EItemDataType
       9: areturn
}
- 是个final类型的,不允许继承自其它类型
 - 继承了java.lang.Enum类,更说明这个枚举就是个class
 
public final class EItemDataType extends java.lang.Enum<EItemDataType> {
- 所有的枚举值都被定义成静态值了,且以常量形式存在
 
public static final EItemDataType Real;
- 再看下一个特殊的方法,由于枚举继承了java.lang.Enum这个类,那么它自然拥有一些实用的方法:
 
public static EItemDataType valueOf(java.lang.String);
这是个字符串参数类型的方法,和我上面定义的valueOf(int value)很像,其目的都是根据一定的条件获取枚举值,只不过方式不同而已,前者是自带的根据枚举值toString的结果来反向获取枚举值,与toString的对应,比如:EItemDataType.Real.toString()它等于“Real”,再调用EItemDataType.valueOf("Reail"),它等于EItemDataType.Real这个值。自定义的valueOf(int value)方式个人感觉并不太好,因为容易与自带的那个方法冲突,最好是改个名称,比如value什么。

最后我们再来看下枚举所能实现的奇葩功能:单例(之前学习.net时写的日记:老生常谈:单件模式)。刚开始看到java的单例可以通过枚举实现时,我都惊呆了,最大的反应是枚举是个存储值的怎么和单例有关系?单例不是class的事吗?其实通过上面的理解,枚举就是个类,那么再想想单例就不会有什么疑问了,把它当成一个普通类不就好了,我们看一个简单的计数的例子:按照上面字节码的结构,这个INSTANCE2会被定义成一个静态变量,正是利用静态变量唯一性的特性来实现了单例,而且是线程安全的。
public enum SafeSingleton implements Serializable {
    INSTANCE2;
    int count;
    public void addCount(int i)
    {
        this.count+=i;
    }
    public void printCount()
    {
        System.out.println(this.count);
    }
}
下面这段程序会输出5050
       for(int i=1;i<=100;i++){
            SafeSingleton.INSTANCE2.addCount(i);
        }
        SafeSingleton.INSTANCE2.printCount();

java中的枚举是一个比较特殊的数据类型,除了具备值存储的能力还拥有class特性,作用范围相比.net要大,但实现更加复杂些。
java枚举与.net中的枚举区别的更多相关文章
- 浅谈java中的枚举类型(转)
		
用法一:常量 在JDK1.5 之前,我们定义常量都是: public static fianl.... .现在好了,有了枚举,可以把相关的常量分组到一个枚举类型里,而且枚举提供了比常量更多的方法. p ...
 - 【译】Java中的枚举
		
前言 译文链接:http://www.programcreek.com/2014/01/java-enum-examples/ Java中的枚举跟其它普通类很像,在其内部包含了一堆预先定义好的对象集合 ...
 - JAVA中的枚举小结
		
枚举 将一组有限集合创建为一种新的类型,集合里面的值可以作为程序组件使用: 枚举基本特性 以下代码是枚举的简单使用: 使用values方法返回enum实例的数组 使用ordinal方法返回每个enum ...
 - 说说Java中的枚举(一)
		
在实际编程中,往往存在着这样的“数据集”,它们的数值在程序中是稳定的,而且“数据集”中的元素是有限的.例如星期一到星期日七个数据元素组成了一周的“数据集”,春夏秋冬四个数据元素组成了四季的“数据集”. ...
 - Java中Enum枚举的使用
		
三种不同的用法 注意项: 1.在switch中使用枚举能使代码的可读性更强. 2.如果要自定义方法,那么必须在enum实例序列的最后添加分号.而且Java要求必须先定义enum实例. 3.所有 ...
 - java实现.net中的枚举
		
Java 和 .net中的枚举不一样,在.net中,枚举是属于值类型的,而在java中确实引用类型的(其实就是一个特殊的类,enum默认集成java.lang.Enum类),所以在java中操作枚举类 ...
 - Java中的枚举类型详解
		
枚举类型介绍 枚举类型(Enumerated Type) 很早就出现在编程语言中,它被用来将一组类似的值包含到一种类型当中.而这种枚举类型的名称则会被定义成独一无二的类型描述符,在这一点上和常量的定义 ...
 - Java中的枚举的治理
		
版权声明:本文为博主原创文章,转载请注明出处,欢迎使劲喷 一.为啥用枚举&为啥要对枚举进行治理 1.先来说说为啥用枚举 表中某个字段标识了这条记录的状态,我们往往使用一些code值来标识,例如 ...
 - JAVA中的枚举类
		
某些情况下一个类的对象是有限而且固定的,例如性别就只有两个类(考虑大众情况).因此这种实例有限而且固定的类,java里面叫枚举类.枚举类的关键字是enum,一些基本的命名规则和文件命名等细节和一般的类 ...
 
随机推荐
- CooMark网页颜色取色表
			
CooMark网页颜色取色表 艾利斯兰 - #F0F8FF - aliceblue 古董白 - #FAEBD7 - antiquewhite 浅绿色 - #00FFFF - aqua 碧绿色 - #7 ...
 - Spring(一)第一个示例
			
原文链接:http://www.orlion.ga/185/ 一.下载与安装Spring 1.访问https://repo.spring.io/webapp/#/artifacts/browse/tr ...
 - Unity3D Player角色移动控制脚本
			
1. 简介 在Unity3D中,有多种方式可以改变物体的坐标,实现移动的目的,其本质是每帧修改物体的position.之前写过类似的文章,这次增加了平时常用API的脚本,每个脚本均手打测试可用. 2. ...
 - HTML5移动Web开发(七)——通过界面图标启动Web应用
			
现在我们要使用手机上某个应用时,通过点击屏幕上的图标就可以运行.但是对基于HTML的Web应用来说,运行起来就比较麻烦了,用户必须先打开浏览器,然后访问想使用的应用程序站点.现在我们想把一个指定的We ...
 - DA - 信息获取途径汇总
			
目的驱动 大多数情况下,都是为了解决某个问题或完成某项任务,才需要进行针对性的.大范围的.细致化的信息获取. 那么,信息获取的方式和来源,就应该紧紧围绕这个"问题和任务"本身来确定 ...
 - Windows Azure Active Directory (1) 前言 - 基于声明的验证和授权
			
<Windows Azure Platform 系列文章目录> 在我们介绍整套系统架构之前,我们需要首先定义一些基本的概念. 用户及其属性: 用户值得是要使用某项服务的个体.用户一般都有一 ...
 - Theano入门神经网络(四)
			
这一部分主要介绍用Theano 实现softmax函数. 在多分类任务中经常用到softmax函数,首先上几个投影片解释一下 假设目标输出是y_hat ,激活函数后的Relu y 一个是1.2 一个是 ...
 - Elasticsearch——Search的基本介绍
			
Elasticsearch最常用的方法莫过于查询了.Es支持以URI请求参数或者请求体的方式进行查询. 查询范例 Elasticsearch支持对多索引以及多类型进行查询. 比如,下面对某个特定索引的 ...
 - SQL Server代理(5/12):理解SQL代理错误日志
			
SQL Server代理是所有实时数据库的核心.代理有很多不明显的用法,因此系统的知识,对于开发人员还是DBA都是有用的.这系列文章会通俗介绍它的很多用法. 如我们在这个系列的前几篇文章所见,SQL ...
 - html5的canvas绘制迷宫地图
			
canvas标签一直是html5的亮点,用它可以实现很多东西.我想用它来绘画像迷宫那样的地图.借助到的工具有瓦片地图编辑器tiled(点击跳转到下载链接). 如图:如果你想要画像这样的迷宫地图,如果不 ...