Java枚举与.net枚举区别详解

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

习惯性观点一:枚举的定义应该与.net相同,比如在.net中我们可以这样定义枚举。

public enum EItemDataType
{
  Real=1,
  Service=2
 }

但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)1;

getValue的方式,明显是需要将一个枚举转换成它所对应的值,.net中也不需要调用方法来取值,也可以强转,比如:

var itemTypeValue=(int)EItemDataType.Real;

私有构造函数,我们可以传多少参数,比如常见的我们需要显示这个枚举值对应的中文描述,在java中我们只需要在构造函数中增加一个name参数就可以了,但在.net中因为没有这货不能这样做,但可以通过 Atrribute来完成。

public enum EItemDataType
{
  [Description("实物")]
  Real=1,
  [Description("服务")]
  Service=2
 }

习惯性观点二:因为.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要大,但实现更加复杂些。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Java的枚举enum示例详解

    一. 什么是枚举 枚举是一种数据类型,具有集合的一些特点,可以存放多个元素,但存储对象有限且固定,枚举也有比较常见的使用场景,如我们需要表达性别(男.女),颜色(红.黄.蓝),星期(星期一.星期二...星期日),四季(春.夏.秋.冬),地理位置(东.西.南.北),方向(前.后.左.右)等,这些场景都非常适合枚举. 二. 定义枚举 java中使用enum来定义枚举,和class,interface具有同样的等级,(注意是enum,而不是Enum),定义枚举可以有两种情况 第一种:默认构造器(空构造

  • Java枚举类接口实例原理解析

    这篇文章主要介绍了Java枚举类接口实例原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 枚举类可以实现一个或多个接口.与普通类实现接口完全一样,枚举类实现接口时,需要实现该接口所包含的方法. 如果需要每个枚举值在调用同一个方法时呈现不同的行为,则可以让每个枚举值在{...}匿名块中实现自己的业务逻辑. public interface IGradeHandler { String getGrade(String studentName)

  • Java包含抽象方法的枚举类示例

    本文实例讲述了Java包含抽象方法的枚举类.分享给大家供大家参考,具体如下: 一 点睛 可以在枚举类里定义一个抽象方法,然后把这个抽象方法交给各枚举值去实现即可. 枚举类里定义抽象方法时无需显式使用abstract关键字将枚举类定义成抽象类,但因为枚举类需要显式创建枚举值,而不是作为父类,所以定义每个枚举值时必须为抽象方法提供实现,否则将出现编译错误. 二 实战 1 代码 public enum Operation { PLUS { public double eval(double x , d

  • Java枚举抽象方法实例解析

    这篇文章主要介绍了Java枚举抽象方法实例解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 需求背景 需求已经确定了几个固定的常量值,并且每个常量值都有相同的行为,但是具体实现细节不同.建议使用枚举抽象方法,优点:结构清晰,便于扩展. 枚举类实现抽象方法 与常规抽象类一样,enum类允许我们为其定义抽象方法,然后使每个枚举实例都实现该方法,以便产生不同的行为方式,注意abstract关键字对于枚举类来说并不是必须的,举个栗子: public

  • Java实现接口的枚举类示例

    本文实例讲述了Java实现接口的枚举类.分享给大家供大家参考,具体如下: 一 点睛 枚举类也可以实现一个或多个接口.与普通类实现一个或多个接口完全一样,枚举类实现一个或多个接口时,也需要实现该接口所包含的方法. 如果需要每个枚举值在调用同一个方法时呈现出不同的行为方式,则可以让每个枚举值分别来实现该方法,每个枚举值提供不同的实现方式,从而让不同枚举值调用同一个方法时具有不同的行为方式. 二 实战 1 代码 GenderDesc.java public interface GenderDesc {

  • java中枚举原来还可以这么用

    前言 相信不少java开发者写过状态变更的业务,比如订单流程.请假流程等等.一般会搞一个状态标识来标识生命周期的某个阶段.很多人会写这种逻辑: 如果流程几十个岂不是要if到爆炸.还有这"0"."1"是几个意思? 优化的办法当然可以使用设计模式中的状态模式来搞,允许一个具有状态的对象根据其状态封装同一对象的不同行为.我们可以编程状态之间的转换,然后定义单独的状态: 但是这种操作会增加过多的状态对象依赖.那么有没骚操作呢?当然有.我们先来了解了解状态机. 状态机 状态机

  • Java手动方式创建枚举类示例

    本文实例讲述了Java手动方式创建枚举类.分享给大家供大家参考,具体如下: 一 点睛 可以采用如下设计方式手动创建枚举类 通过private将构造器隐藏起来. 把这个类的所有可能实例都使用public static final属性来保存. 如果有必要,可以提供一些静态方法,允许其他程序根据特定参数来获取与之匹配实例. 二 代码 1 Season.java public class Season { // 把Season类定义成不可变的,将其成员变量也定义成final的 private final

  • java枚举类的属性、方法和构造方法应用实战

    本文实例讲述了java枚举类的属性.方法和构造方法应用.分享给大家供大家参考,具体如下: 一 点睛 枚举类也是一种类,只是它是一种比较特殊的类,因此它一样可以使用属性和方法. 枚举类通常应该设计成不可变类,也就说它的属性值不应该允许改变,这样会更安全,而且代码更加简洁.为此,我们应该将枚举类的属性都使用private final修饰. 一旦为枚举类显式定义了带参数的构造器,则列出枚举值时也必须对应地传入参数. 二 简单枚举类实战 1 代码 Gender.java public enum Gend

  • Java中 % 与Math.floorMod() 区别详解

    %为取余(rem),Math.floorMod()为取模(mod) 取余取模有什么区别呢? 对于整型数a,b来说,取模运算或者取余运算的方法都是: 1.求 整数商: c = a/b; 2.计算模或者余数: r = a - c*b. 区别是: 取余运算在计算商值向0方向舍弃小数位 取模运算在计算商值向负无穷方向舍弃小数位 比如a=4,b=-3时,a/b = -1.3333... 此时,取余c=1,取模c=-2 (%在不同语言中有不同的意义,比如Java或者c/c++中%为取余,python中%则为

  • java中“==“和equals()的区别详解

    今天我们探讨一下Java中"=="与equals()的区别 ==:关系运算符 在基本数据类型中比较两个值的内容是否相等       在引用类型型中比较的是两个对象的地址是否相等 equals()是Object类中的方法 1.基本数据类型无法使用equals()方法 2.在引用类型中若是没有重写Object类时,则默认使用Object类的equals方法,则仍然 利用"=="比较两个对象的内存地址,若是重写Object类的equals方法,则调用子类重写后    的方

  • Java 接口和抽象类的区别详解

    什么是抽象类和接口? 区别在哪里? 不同的编程语言对接口和抽象类的定义方式可能有些差别,但是差别并不大.本文使用 Java 语言. 抽象类 下面我们通过一个例子来看一个典型的抽象类的使用场景. Logger 是一个记录日志的抽象类,FileLogger 和 MessageQueueLogger 继承Logger,分别实现两种不同的日志记录方式: 记录日志到文件中 记录日志到消息队列中 FileLogger 和 MessageQueuLogger 两个子类复用了父类 Logger 中的name.e

  • java HashMap和HashTable的区别详解

    HashMap和HashTable,这二者的区别经常被别人问起,今天在此总结一下. (一)继承的历史不同 public class Hashtable extends Dictionary implements Map public class HashMap extends AbstractMap implements Map Hashtable是继承自Dictionary类的,而HashMap则是Java 1.2引进的Map接口的一个实现. (二)安全性不同 HashMap是非synchro

  • java基础之 “==”与“equals”区别详解

    对于初学java的人来说,在面对数值比较的时候,我们大多数会采用 "=="的方式来进行比较,但是java中给我们提供了equals()方法,这时候很多人就会忽略这两种方式的区别,在学习中产生了很多错误,本文将详细区分equals和 == 两种方式的区别. "==" 解读 对于基本类型和引用类型,==的作用效果是不同的,对于 基本类型 来说,比较的是值是否相同,对于 引用类型 来说,比较的是引用是否相同. 代码示例: public static void main(S

  • java  HashMap和HashTable的区别详解

    HashMap和HashTable,这二者的区别经常被别人问起,今天在此总结一下. (一)继承的历史不同 public class Hashtable extends Dictionary implements Map public class HashMap extends AbstractMap implements Map Hashtable是继承自Dictionary类的,而HashMap则是Java 1.2引进的Map接口的一个实现. (二)安全性不同 HashMap是非synchro

  • java ArrayList和Vector的区别详解

     ArrayList和Vector的区别 相同点: 1.ArrayList和Vector都是继承了相同的父类和实现了相同的接口 2.底层都是数组实现的 3.初始默认长度都为10. 不同点: 1.同步性: Vector中的public方法多数添加了synchronized关键字,以确保方法同步,也即是Vector线程安全,ArrayList线程不安全. 2.扩容不同 内部属性不同,这可能是导致扩容方式不同的原因所在. ArrayList有两个属性,存储数据的数组elementData,和存储记录数

  • JAVA序列化Serializable及Externalizable区别详解

    序列化简介 Java 的对象序列化将那些实现 Serializable 接口的对象转换成一个字节序列,并能在之后将这个字节序列完全恢复为原来的对象. 这就意味着 Java 对象在网络上的传输可以不依赖于当前计算机的操作系统,就可以将对象进行传递,这也是Java跨平台的一种体现. Java 对象的序列化主要支持两种特性: 1.Java的远程方法调用(Remote Method Invocation RMI): 2.对于 JavaBean 来说,序列化也是必须的. 要序列化一个对象,需要创建一个 O

  • Java Comparable及Comparator接口区别详解

    在实际应用中,我们往往有需要比较两个自定义对象大小的地方.而这些自定义对象的比较,就不像简单的整型数据那么简单,它们往往包含有许多的属性,我们一般都是根据这些属性对自定义对象进行比较的.所以Java中要比较对象的大小或者要对对象的集合进行排序,需要通过比较这些对象的某些属性的大小来确定它们之间的大小关系. 一般,Java中通过接口实现两个对象的比较,比较常用就是Comparable接口和Comparator接口.首先类要实现接口,并且使用泛型规定要进行比较的对象所属的类,然后类实现了接口后,还需

  • Java中equals和==的区别详解

    目录 1.java中的数据类型,可分为两类: 2.再稍微改动一下程序,会有更奇怪的发现: 3. 字符串缓冲池 4.再次更改程序: 总结 1.java中的数据类型,可分为两类: 1.基本数据类型,也称原始数据类型. byte,short,char,int,long,float,double,boolean 他们之间的比较,应用双等号(==),比较的是他们的值. 2.复合数据类型(类) 当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址,所以,除非是同一个new出来的对象,他们的比较后的

随机推荐