亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

必須了解的高階JAVA枚舉特性!

 更新時間:2021年01月13日 11:27:55   作者:程序猿DD  
這篇文章主要介紹了必須了解的高階JAVA枚舉特性!幫助大家更好的理解和學習Java枚舉的相關(guān)知識,感興趣的朋友可以了解下

JAVA枚舉,比你想象中還要有用!

我經(jīng)常發(fā)現(xiàn)自己在Java中使用枚舉來表示某個對象的一組潛在值。

在編譯時確定類型可以具有什么值的能力是一種強大的能力,它為代碼提供了結(jié)構(gòu)和意義。

當我第一次了解枚舉時,當時我認為它們只是一個為常量命名的工具,可以很容易地被靜態(tài)常量字符串ENUM_VAL_NAME所取代。

后來我發(fā)現(xiàn)我錯了。事實證明,Java枚舉具有相當高級的特性,可以使代碼干凈、不易出錯,功能強大。

讓我們一起來看看Java中的一些高級枚舉特性,以及如何利用這些特性使代碼更簡單、更可讀。

枚舉是類!

在Java中,枚舉是Object的一個子類。讓我們看看所有枚舉的基類,Enum(為簡潔起見進行了修改)。

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;
 }
}

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

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

我們能夠為枚舉提供實例方法、構(gòu)造函數(shù)和字段。我們可以重寫toString(),但不能重寫hashCode()或equals(Object other)。

接下來我們看下我們的枚舉示例,Operation

 enum Operation {
  ADD,
  SUBTRACT,
  MULTIPLY
 }

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

 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();
 }
}

當然,這樣子會有一些問題。

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

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

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

這是因為Java編譯器知道上面的第一個問題,并且希望確保我們能夠處理在不知情的情況下向Operation中添加了新枚舉。

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

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

因為枚舉是類,所以我們可以創(chuàng)建一個枚舉字段來保存執(zhí)行操作的函數(shù)。

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

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

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

因為我們現(xiàn)在從Operation中調(diào)用方法,所以我們可以將其更改為實例方法并使用this,而不是用Operation.apply()來實現(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);

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

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);
       }
 
 }

這里我做的是:

  • 添加了一個字段 BiFunction<Integer, Integer, Integer> operation
  • 用BiFunction創(chuàng)建了用于Operation的構(gòu)造函數(shù)。
  • 調(diào)用枚舉定義中的構(gòu)造函數(shù),并用lambda指定BiFunction<Integer, Integer, Integer>。

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

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

因為BiFunction是用@functioninterface注釋的,所以我們可以使用Lambda表示法定義一個。

因為我們的函數(shù)接受兩個參數(shù),所以我們可以使用(x,y)來指定它們。

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

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

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

但是,這種實現(xiàn)更好,因為編譯器會告訴我們何時添加了新Operation,這要求我們更新新函數(shù)。如果沒有這一點,如果我們在添加新Operation時還不記得更新switch語句,就有可能得到UnsupportedOperationException()。

關(guān)鍵要點

  • Enum枚舉是Enum的擴展類。
  • Enum枚舉可以有字段、構(gòu)造函數(shù)和實例方法。
  • Enum枚舉字段可以存儲函數(shù)。與lambdas配合使用,可以創(chuàng)建干凈、安全的特定于枚舉的函數(shù)實現(xiàn),并在編譯時強制執(zhí)行它們(而不是使用switch)。

下面是這個示例的GitHub地址。(https://github.com/alex-power/java-enum-example

本文參考:https://medium.com/javarevisited/advanced-java-enum-features-you-need-to-know-b516a191c7e2

以上就是必須了解的高階JAVA枚舉特性!的詳細內(nèi)容,更多關(guān)于高階JAVA枚舉特性的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Java中的ConcurrentLinkedQueue松散隊列解析

    Java中的ConcurrentLinkedQueue松散隊列解析

    這篇文章主要介紹了Java中的ConcurrentLinkedQueue松散隊列解析,鏈表是松散的,鏈表節(jié)點并不都是有效的,允許存在無效節(jié)點val=null,但是只有最后一個節(jié)點才能next=null,需要的朋友可以參考下
    2023-12-12
  • Win10系統(tǒng)下配置java環(huán)境變量的全過程

    Win10系統(tǒng)下配置java環(huán)境變量的全過程

    這篇文章主要給大家介紹了關(guān)于Win10系統(tǒng)下配置java環(huán)境變量的相關(guān)資料,文中通過圖文介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-11-11
  • 深入理解happens-before和as-if-serial語義

    深入理解happens-before和as-if-serial語義

    本文大部分整理自《Java并發(fā)編程的藝術(shù)》,溫故而知新,加深對基礎(chǔ)的理解程度。下面可以和小編來一起學習下
    2019-05-05
  • Java深入淺出數(shù)組的定義與使用下篇

    Java深入淺出數(shù)組的定義與使用下篇

    數(shù)組是有序的元素序列,若將有限個類型相同的變量的集合命名,那么這個名稱為數(shù)組名。組成數(shù)組的各個變量稱為數(shù)組的分量,也稱為數(shù)組的元素,有時也稱為下標變量。數(shù)組是在程序設(shè)計中,為了處理方便,?把具有相同類型的若干元素按有序的形式組織起來的一種形式
    2022-03-03
  • IDEA中GitLab的使用詳解

    IDEA中GitLab的使用詳解

    這篇文章主要介紹了IDEA中GitLab的使用,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-07-07
  • Java實現(xiàn)對象列表導出為excel表格的實用工具類

    Java實現(xiàn)對象列表導出為excel表格的實用工具類

    這篇文章主要為大家詳細介紹了Java如何實現(xiàn)對象列表導出為excel表格的實用工具類,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下
    2023-12-12
  • Java多線程之定時器Timer的實現(xiàn)

    Java多線程之定時器Timer的實現(xiàn)

    定時/計劃功能在Java應用的各個領(lǐng)域都使用得非常多,比方說Web層面。本文主要為大家介紹了Java多線程中定時器Timer的實現(xiàn),感興趣的小伙伴可以了解一下
    2022-10-10
  • Java中WeakHashMap和HashMap的區(qū)別詳解

    Java中WeakHashMap和HashMap的區(qū)別詳解

    這篇文章主要介紹了Java中WeakHashMap和HashMap的區(qū)別詳解,WeakHashMap和HashMap一樣,WeakHashMap也是一個散列表,它存儲的內(nèi)容也是鍵值對(key-value)映射,而且鍵和值都可以為null,需要的朋友可以參考下
    2023-09-09
  • java利用delayedQueue實現(xiàn)本地的延遲隊列

    java利用delayedQueue實現(xiàn)本地的延遲隊列

    這篇文章主要給大家介紹了java利用delayedQueue實現(xiàn)本地的延遲隊列的相關(guān)資料,文中介紹的非常詳細,相信對大家具有一定的參考價值,需要的朋友們下面來一起看看吧。
    2017-04-04
  • Java日期時間字符串和毫秒相互轉(zhuǎn)換的方法

    Java日期時間字符串和毫秒相互轉(zhuǎn)換的方法

    這篇文章主要為大家詳細介紹了Java日期時間字符串和毫秒相互轉(zhuǎn)換的方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-12-12

最新評論