溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶(hù)服務(wù)條款》

JAVA中枚舉的特性有哪些

發(fā)布時(shí)間:2021-01-13 14:04:41 來(lái)源:億速云 閱讀:151 作者:Leah 欄目:開(kāi)發(fā)技術(shù)

本篇文章給大家分享的是有關(guān)JAVA中枚舉的特性有哪些,小編覺(jué)得挺實(shí)用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話不多說(shuō),跟著小編一起來(lái)看看吧。

枚舉是類(lèi)!

在Java中,枚舉是Object的一個(gè)子類(lèi)。讓我們看看所有枚舉的基類(lèi),Enum(為簡(jiǎn)潔起見(jiàn)進(jìn)行了修改)。

public abstract class Enum<E extends Enum<E>>
  implements Constable, Comparable<E>, Serializable {
 private final String name;
 
 public final String name() {
   return name;
 }
 
 private final int ordinal;
 
 public final int ordinal() {
   return ordinal;
 }
 
 protected Enum(String name, int ordinal) {
   this.name = name;
   this.ordinal = ordinal;
 }
 
 public String toString() {
   return name;
 }
 
 public final boolean equals(Object other) {
   return this==other;
 }
 
 public final int hashCode() {
   return super.hashCode();
 }
 
 public final int compareTo(E o) {
   Enum<?> other = (Enum<?>)o;
   Enum<E> self = this;
   if (self.getClass() != other.getClass() && // optimization
     self.getDeclaringClass() != other.getDeclaringClass())
     throw new ClassCastException();
   return self.ordinal - other.ordinal;
 }
}

我們可以看到,這基本上只是一個(gè)常規(guī)的抽象類(lèi),有兩個(gè)字段,name和ordinal。

所以說(shuō)枚舉都是類(lèi),所以它們具有常規(guī)類(lèi)的許多特性。

我們能夠?yàn)槊杜e提供實(shí)例方法、構(gòu)造函數(shù)和字段。我們可以重寫(xiě)toString(),但不能重寫(xiě)hashCode()或equals(Object other)。

接下來(lái)我們看下我們的枚舉示例,Operation

 enum Operation {
  ADD,
  SUBTRACT,
  MULTIPLY
 }

這個(gè)枚舉表示一個(gè)Operation可以對(duì)兩個(gè)值執(zhí)行,并將生成一個(gè)結(jié)果。關(guān)于如何實(shí)現(xiàn)此功能,您最初的想法可能是使用switch語(yǔ)句,如下所示:

 public int apply(Operation operation, int arg1, int arg2) {
  switch(operation) {
   case ADD:
    return arg1 + arg2;
   case SUBTRACT:
    return arg1 - arg2;
   case MULTIPLY:
    return arg1 * arg2;
   default:
    throw new UnsupportedOperationException();
 }
}

當(dāng)然,這樣子會(huì)有一些問(wèn)題。

第一個(gè)問(wèn)題是,如果我們將一個(gè)新操作添加到我們的枚舉Operation中,編譯器不會(huì)通知我們這個(gè)開(kāi)關(guān)不能正確處理新操作。

更糟糕的是,如果一個(gè)懶惰的開(kāi)發(fā)人員在另一個(gè)類(lèi)中復(fù)制或重新編寫(xiě)這些代碼,我們可能無(wú)法更新它。

第二個(gè)問(wèn)題是默認(rèn)情況default,每段程序里面都是必需的,盡管我們知道在正確的代碼里它永遠(yuǎn)不會(huì)發(fā)生。

這是因?yàn)镴ava編譯器知道上面的第一個(gè)問(wèn)題,并且希望確保我們能夠處理在不知情的情況下向Operation中添加了新枚舉。

還好,Java8用函數(shù)式編程為我們提供了一個(gè)干凈的解決方案。

函數(shù)枚舉實(shí)現(xiàn)

因?yàn)槊杜e是類(lèi),所以我們可以創(chuàng)建一個(gè)枚舉字段來(lái)保存執(zhí)行操作的函數(shù)。

但是在我們找到解決方案之前,讓我們先來(lái)看看一些重構(gòu)。

首先,讓我們把開(kāi)關(guān)放在enum類(lèi)中。

enum Operation {
 ADD,
 SUBTRACT,
 MULTIPLY;
 
 public static int apply(Operation operation, int arg1, int arg2) {
  switch(operation) {
   case ADD:
    return arg1 + arg2;
   case SUBTRACT:
    return arg1 - arg2;
   case MULTIPLY:
    return arg1 * arg2;
   default:
    throw new UnsupportedOperationException();
  }
 }
}

我們可以這樣做:Operation.apply(Operation.ADD, 2, 3);

因?yàn)槲覀儸F(xiàn)在從Operation中調(diào)用方法,所以我們可以將其更改為實(shí)例方法并使用this,而不是用Operation.apply()來(lái)實(shí)現(xiàn),如下所示:

public int apply(int arg1, int arg2) {
 switch(this) {
  case ADD:
   return arg1 + arg2;
  case SUBTRACT:
   return arg1 - arg2;
  case MULTIPLY:
   return arg1 * arg2;
  default:
   throw new UnsupportedOperationException();
 }
}

像這樣使用:Operation.ADD.apply(2, 3);

看起來(lái)變好了?,F(xiàn)在讓我們更進(jìn)一步,通過(guò)使用函數(shù)式編程完全消除switch語(yǔ)句。

enum Operation {
       ADD((x, y) -> x + y),
       SUBTRACT((x, y) -> x - y),
       MULTIPLY((x, y) -> x * y);
 
       Operation(BiFunction<Integer, Integer, Integer> operation) {
           this.operation = operation;
       }
 
       private final BiFunction<Integer, Integer, Integer> operation;
 
       public int apply(int x, int y) {
           return operation.apply(x, y);
       }
 
 }

這里我做的是:

  • 添加了一個(gè)字段 BiFunction<Integer, Integer, Integer> operation

  • 用BiFunction創(chuàng)建了用于Operation的構(gòu)造函數(shù)。

  • 調(diào)用枚舉定義中的構(gòu)造函數(shù),并用lambda指定BiFunction<Integer, Integer, Integer>。

這個(gè)java.util.function.BiFunction operation字段是對(duì)采用兩個(gè)參數(shù)的函數(shù)(方法)的引用。

在我們的例子中,兩個(gè)參數(shù)都是int型,返回值也是int型。不幸的是,Java參數(shù)化類(lèi)型不支持原語(yǔ),所以我們必須使用Integer。

因?yàn)锽iFunction是用@functioninterface注釋的,所以我們可以使用Lambda表示法定義一個(gè)。

因?yàn)槲覀兊暮瘮?shù)接受兩個(gè)參數(shù),所以我們可以使用(x,y)來(lái)指定它們。

然后我們定義了一個(gè)單行方法,它使用 ->x+y 返回一個(gè)值。這相當(dāng)于下面的方法,只是更簡(jiǎn)潔而已。

 class Adder implements BiFunction<Integer, Integer, Integer> {
     @Override
     public Integer apply(Integer x, Integer y) {
         return x + y;
  }
 }

我們的新Operation實(shí)現(xiàn)采用相同的方式:Operation.ADD.apply(2, 3);.

但是,這種實(shí)現(xiàn)更好,因?yàn)榫幾g器會(huì)告訴我們何時(shí)添加了新Operation,這要求我們更新新函數(shù)。如果沒(méi)有這一點(diǎn),如果我們?cè)谔砑有翺peration時(shí)還不記得更新switch語(yǔ)句,就有可能得到UnsupportedOperationException()。

關(guān)鍵要點(diǎn)

  • Enum枚舉是Enum的擴(kuò)展類(lèi)。

  • Enum枚舉可以有字段、構(gòu)造函數(shù)和實(shí)例方法。

  • Enum枚舉字段可以存儲(chǔ)函數(shù)。與lambdas配合使用,可以創(chuàng)建干凈、安全的特定于枚舉的函數(shù)實(shí)現(xiàn),并在編譯時(shí)強(qiáng)制執(zhí)行它們(而不是使用switch)。

以上就是JAVA中枚舉的特性有哪些,小編相信有部分知識(shí)點(diǎn)可能是我們?nèi)粘9ぷ鲿?huì)見(jiàn)到或用到的。希望你能通過(guò)這篇文章學(xué)到更多知識(shí)。更多詳情敬請(qǐng)關(guān)注億速云行業(yè)資訊頻道。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI