Java枚举是什么?

2021-06-23 19:47发布

15条回答
打了代码淦程序
1楼 · 2021-06-24 15:44.采纳回答

概念: 枚举是和类以及接口同一个级别的类型(都是引用类型的数据),可以理解为它是一种特殊类的体现,本质就是 (枚举类型编译后也会生成字节码文件). 特殊:枚举是对只有有限对象个数的类的独立(枚举类型中提供对象个数是在定义枚举的时候就已经决定,外面 不能添加枚举对象,在外面只能获取枚举对象使用), 比如:星期只能有7个对象 月份类对应的对象个数12..... 枚举有自己的关键字: enum

特点:v>

1. 所有枚举类都是Enum的子类,Enum这是所有 Java 语言枚举类型的公共基本类。

2.v> 我们可以通过"枚举类名.枚举项名称"去访问指定的枚举项.

3.v> 每一个枚举项其实就是该枚举的一个对象【默认都是被public static final修饰的,不能显示的添加 修饰符】

4.v> 枚举也是一个类,也可以去定义成员变量 成员方法 抽象方法

5.v> 枚举类的第一行上必须是枚举项,最后一个枚举项后的分号是可以省略的,但是如果枚举类有其他的东 西,这个分号 就不能省略。建议不要省略

6.v> 枚举类可以有构造器,但必须是private的,它默认的也是private. 枚举项的用法比较特殊:枚举项(实参);

7.v>  枚举类也可以有抽象方法,但是枚举项必须重写该方法


不吃鱼的猫
2楼 · 2021-06-24 09:50

我们学习过单例模式,即一个类只有一个实例。而枚举其实就是多例,一个类有多个实例,但实例的个数不是无穷的,是有限个数的

Java 枚举是一个特殊的类,一般表示一组常量,比如一年的 4 个季节,一个年的 12 个月份,一个星期的 7 天,方向有东南西北等。

Java 枚举类使用 enum 关键字来定义,各个常量使用逗号 , 来分割。


枚举,是java中的一种比较特殊的class,它继承了java.lang.Enum类,也就是说在枚举类中是不能够继承其他类的(java单根继承,將枚举类的.class文件反编译,就可以看到它继承了哪个类了。),但可以实现接口。枚举更像是一个有限常量的集合,类似于数学中的A={1,2,3,4}。当你使用枚举时,只能使用枚举内已经存在的。枚举内的元素默认被public static final修饰,是不是很眼熟?像是常量?在枚举中,每一个元素都是有自己的序号的,从0开始,我们可以通过元素名称来确定枚举类中元素的位置,也可以通过枚举类中元素的序号来确定元素的位置。


Uzi
5楼 · 2021-06-25 18:38

我们使用enum定义的枚举类型,会在编译之后转化为一个继承了java.lang.Enum的类,而我们定义的每个枚举值都会在类的初始化阶段被实例化为我们所定义的枚举类的一个对象。同时,编译器还帮我们在类中增加了两个方法,分别是:values()和valueOf()。


寂静的枫林
6楼 · 2021-06-26 10:14

枚举,是java中的一种比较特殊的class,它继承了java.lang.Enum类,也就是说在枚举类中是不能够继承其他类的(java单根继承,將枚举类的.class文件反编译,就可以看到它继承了哪个类了。),但可以实现接口。

yjh
7楼 · 2021-06-29 16:48

java枚举主要是定义一种拥有多个相关值的数据类型。可以把它当成一个统一取静态常量的类,方便维护项目静态常量

ZGSYT
8楼 · 2021-07-02 10:31

枚举其实就是多例

我是大脸猫
9楼 · 2021-07-02 14:24

枚举类型是什么呢,其实是指一组固定的常量组成合法值的类型。

java中如何定义,看下面代码:


public enum EnumDemo {

    A, B

}

这就是枚举类型。我们知道,java面向对象的语言,不就是对象和类吗,这enum是什么东西?其实这是java语法,是编译器优化,看下面,javap后的汇编:


public final class com.common.EnumDemo extends java.lang.Enum {

  public static final com.common.EnumDemo A;         //2个本类的实例,并且是静态final域


  public static final com.common.EnumDemo B;


  static {};

    Code:

       0: new           #1                  // class com/common/EnumDemo

       3: dup           

       4: ldc           #13                 // String A

       6: iconst_0                          //下面是调用父类构造方法进行初始化

       7: invokespecial #14                 // Method "":(Ljava/lang/String;I)V

      10: putstatic     #18                 // Field A:Lcom/common/EnumDemo;//赋值给静态域A

      13: new           #1                  // class com/common/EnumDemo  下面同样赋值静态域B

      16: dup           

      17: ldc           #20                 // String B

      19: iconst_1      

      20: invokespecial #14                 // Method "":(Ljava/lang/String;I)V

      23: putstatic     #21                 // Field B:Lcom/common/EnumDemo;

      26: iconst_2                           //常量2入栈,实例化数组

      27: anewarray     #1                  // class com/common/EnumDemo

      30: dup           

      31: iconst_0      

      32: getstatic     #18                 // Field A:Lcom/common/EnumDemo;

      35: aastore                           //把A赋值给数组第一个元素,下面的是赋值个第二个元素

      36: dup           

      37: iconst_1      

      38: getstatic     #21                 // Field B:Lcom/common/EnumDemo;

      41: aastore       

      42: putstatic     #23                 // Field ENUM$VALUES:[Lcom/common/EnumDemo;

      45: return        


public static com.common.EnumDemo[] values(); //此方法是获取静态代码块中生成的枚举类型数组

    Code:

       0: getstatic     #23                 // Field ENUM$VALUES:[Lcom/common/EnumDemo;

       3: dup           

       4: astore_0      

       5: iconst_0      

       6: aload_0       

       7: arraylength   

       8: dup           

       9: istore_1      

      10: anewarray     #1                  // class com/common/EnumDemo

      13: dup           

      14: astore_2      

      15: iconst_0      

      16: iload_1       

      17: invokestatic  #31                 // Method java/lang/System.arraycopy:(Ljava/lang/Object;ILjava/lang/Object;II)V

      20: aload_2       

      21: areturn       


  public static com.common.EnumDemo valueOf(java.lang.String);

    Code:

       0: ldc           #1                  // class com/common/EnumDemo

       2: aload_0       

       3: invokestatic  #39                 // Method java/lang/Enum.valueOf:(Ljava/lang/Class;Ljava/lang/String;)Ljava/lang/Enum;

       6: checkcast     #1                  // class com/common/EnumDemo

       9: areturn       

}


发现没,其实枚举也是类,并且还继承了java.lang.Enum。由汇编可以看出,枚举的初始化过程是在静态代码块中进行,我们知道,静态代码块只在类首次加载的时候执行一次,final域保证引用地址不会改变。同时,我们也没有看到有公有的构造方法,客户端也就不能创建新的实例,由此可见,枚举类型具有实例受限的特性。

实例受限,很容易让我们想到一种设计模式。没错,就是单例。接下来我们我如何用枚举实现单例。


public enum EnumDemo {

    A;

    public void doSomeThing() {

        System.out.println("dosomething");

    }


    public static void main(String[] args) {

        EnumDemo.A.doSomeThing();

    }

}


看代码,就是这么简单,实例唯一,就是单例,什么线程安全,什么双重检验,什么懒汉饿汉模式,都不用考虑,枚举天然的适合做为单例。这也是Effective Java推崇的做法。

可能有些人用枚举来表示状态,但是需要用数字表示,好存入数据库,这个也简单,只需要为这个类加个属性就ok了,看代码。


public enum EnumDemo {

    A(2), B(4);


    private int status;


    private EnumDemo(int status) {

        this.status = status;

    }


    public int getStatus() {

        return status;

    }


    public static void main(String[] args) {

        System.out.println(EnumDemo.A.getStatus());

        System.out.println(EnumDemo.B.getStatus());

    }

}


由此可见,用枚举来做常量,会非常直观,也非常的安全,因为有类型安全校验的保证,不会如同int类型的枚举一样,随便一个数字都能进行常量替换。这种怪异的写法,确实不好理解,尤其是有额外的属性后,这就需要熟能生巧了,自己javap慢慢研究。


相关问题推荐

  • 回答 2

    Statement的execute(String query)方法用来执行任意的SQL查询,如果查询的结果是一个ResultSet,这个方法就返回true。如果结果不是ResultSet,比如insert或者update查询,它就会返回false。我们可以通过它的getResultSet方法来获取ResultSet,或者通过getUpda...

  • 回答 22

    忙的时候项目期肯定要加班 但是每天加班应该还不至于

  • 回答 108
    已采纳

    虽然Java人才越来越多,但是人才缺口也是很大的,我国对JAVA工程师的需求是所有软件工程师当中需求大的,达到全部需求量的60%-70%,所以Java市场在短时间内不可能饱和。其次,Java市场不断变化,人才需求也会不断增加。马云说过,未来的制造业要的不是石油,...

  • 回答 5
    已采纳

    工信部证书含金量较高。工信部是国务院的下属结构,具有发放资质、证书的资格。其所发放的证书具有较强的权威性,在全国范围内收到认可,含金量通常都比较高。 工信部证书,其含义也就是工信部颁发并承认的某项技能证书,是具有法律效力的,并且是国家认可的...

  • 回答 70
    已采纳

    学Java好不好找工作?看学完Java后能做些什么吧。一、大数据技术Hadoop以及其他大数据处理技术都是用Java或者其他,例如Apache的基于Java 的 HBase和Accumulo以及ElasticSearchas。但是Java在此领域并未占太大空间,但只要Hadoop和ElasticSearchas能够成长壮...

  • 回答 16
    已采纳

    就是java的基础知识啊,比如Java 集合框架;Java 多线程;线程的五种状态;Java 虚拟机;MySQL (InnoDB);Spring 相关;计算机网络;MQ 消息队列诸如此类

  • 回答 12

    #{}和${}这两个语法是为了动态传递参数而存在的,是Mybatis实现动态SQL的基础,总体上他们的作用是一致的(为了动态传参),但是在编译过程、是否自动加单引号、安全性、使用场景等方面有很多不同,下面详细比较两者间的区别:1.#{} 是 占位符 :动态解析 ...

  • 回答 62

    没问题的,专科学历也能学习Java开发的,主要看自己感不感兴趣,只要认真学,市面上的培训机构不少都是零基础课程,能跟得上,或是自己先找些资料学习一下。

  • 回答 4

    1、反射对单例模式的破坏采用反射的方式另辟蹊径实例了该类,导致程序中会存在不止一个实例。解决方案其思想就是采用一个全局变量,来标记是否已经实例化过了,如果已经实例化过了,第 二次实例化的时候,抛出异常2、clone()对单例模式的破坏当需要实现单例的...

  • 回答 5

     优点: 一、实例控制  单例模式会阻止其他对象实例化其自己的单例对象的副本,从而确保所有对象都访问唯一实例。 二、灵活性  因为类控制了实例化过程,所以类可以灵活更改实例化过程。 缺点: 一、开销  虽然数量很少,但如果每次对象请求引用时都要...

  • 回答 4

    这个主要是看你数组的长度是多少, 比如之前写过的一个程序有个数组存的是各个客户端的ip地址:string clientIp[4]={XXX, xxx, xxx, xxx};这个时候如果想把hash值对应到上面四个地址的话,就应该对4取余,这个时候p就应该为4...

  • 回答 6

     哈希表的大小 · 关键字的分布情况 · 记录的查找频率 1.直接寻址法:取关键字或关键字的某个线性函数值为散列地址。即H(key)=key或H(key) = a·key + b,其中a和b为常数(这种散列函数叫做自身函数)。...

  • 回答 6

    哈希表的大小取决于一组质数,原因是在hash函数中,你要用这些质数来做模运算(%)。而分析发现,如果不是用质数来做模运算的话,很多生活中的数据分布,会集中在某些点上。所以这里最后采用了质数做模的除数。 因为用质数做了模的除数,自然存储空间的大小也用质数了...

  • 回答 2

    是啊,哈希函数的设计至关重要,好的哈希函数会尽可能地保证计算简单和散列地址分布均匀,但是,我们需要清楚的是,数组是一块连续的固定长度的内存空间

  • 回答 3

     解码查表优化算法,seo优化

  • 回答 5

    1.对对象元素中的关键字(对象中的特有数据),进行哈希算法的运算,并得出一个具体的算法值,这个值 称为哈希值。2.哈希值就是这个元素的位置。3.如果哈希值出现冲突,再次判断这个关键字对应的对象是否相同。如果对象相同,就不存储,因为元素重复。如果对象不同,就...

没有解决我的问题,去提问