成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

語法糖甜不甜?巧用枚舉實現“狀態”轉換限制

開發 架構
本質上,JVM 并不支持語法糖,語法糖只存在于編譯期。當編譯器將 .java 源文件編譯成 .class 字節碼文件時,會進行解語法糖的操作,來還原最原始的基礎語法結構。

[[423672]]

語法糖

語法糖(Syntactic sugar),也被譯為糖衣語法,是由英國計算機科學家彼得·約翰·蘭達(Peter J. Landin)發明的一個術語,指計算機語言中添加的某種語法,這種語法對語言的功能并沒有影響,但是更方便程序員使用。通常來說使用語法糖能夠增加程序的可讀性,從而減少程序代碼出錯的機會。——摘抄自百度百科

本質上,JVM 并不支持語法糖,語法糖只存在于編譯期。當編譯器將 .java 源文件編譯成 .class 字節碼文件時,會進行解語法糖的操作,來還原最原始的基礎語法結構。

我們所熟悉的編程語言中幾乎都會包含語法糖,當然 JAVA 也不例外。JAVA 中的語法糖包含條件編譯、斷言、switch 支持 String 與枚舉、可變參數、自動裝箱/拆箱、枚舉、內部類、泛型擦除、增強for循環、lambda表達式、try-with-resources等等。今天我們先來了解下枚舉。

枚舉類

JDK5 提供了一種新的特殊的類——枚舉類,一般在類對象有限且固定的場景下使用,用來替代類中定義常量的方式。枚舉相較于常量更加直觀且類型安全。

枚舉類的使用非常簡單,用 enum 關鍵字來定義,多個枚舉變量直接用逗號隔開。我們先來定義一個簡單的枚舉類 OrderStatus.java

  1. public enum OrderStatus { 
  2.     //未支付、已支付、退款中、退款成功、退款失敗; 
  3.     NO_PAY, PAY, REFUNDING, REFUNDED, FAIL_REFUNDED, ; 

在其他類中使用 enum 變量的時候,只需要【類名.變量名】就可以了,和使用靜態變量一樣。另外,枚舉類型可以確保 JVM 中僅存在一個常量實例,所以我們可以放心的使用“ ==”來比較兩個變量。

注意事項:

枚舉類的第一行必須是枚舉項,最后一個枚舉項后的分號是可以省略的,但是如果枚舉類有其它的東西,這個分號就不能省略。建議不要省略!

枚舉變量最好大寫,多個單詞之間使用”_”隔開(比如:NO_PAY)。

反編譯

我們可以先通過 javac 命令或者 IDEA 的編譯功能將OrderStatus.java 編譯為OrderStatus.class 字節碼文件,然后用DJ Java Decompiler 反編譯器對 .class 文件進行反編譯。

如果需要 DJ Java Decompiler 反編譯器的小伙伴可以私信阿Q獲取!

  1. public final class OrderStatus extends Enum 
  2.  
  3.     //該方法會返回包括所有枚舉變量的數組,可以方便的用來做循環。 
  4.     public static OrderStatus[] values() 
  5.     { 
  6.         return (OrderStatus[])$VALUES.clone(); 
  7.     } 
  8.  
  9.     //根據傳入的字符串,轉變為對應的枚舉變量。 
  10.     //前提是傳的字符串和定義枚舉變量的字符串一抹一樣,區分大小寫。 
  11.     //如果傳了一個不存在的字符串,那么會拋出異常。 
  12.     public static OrderStatus valueOf(String name
  13.     { 
  14.         return (OrderStatus)Enum.valueOf(com/itcast/java/enumpack/OrderStatus, name); 
  15.     } 
  16.  
  17.     private OrderStatus(String s, int i) 
  18.     { 
  19.         super(s, i); 
  20.     } 
  21.  
  22.     public static final OrderStatus NO_PAY; 
  23.     public static final OrderStatus PAY; 
  24.     public static final OrderStatus REFUNDING; 
  25.     public static final OrderStatus REFUNDED; 
  26.     public static final OrderStatus FAIL_REFUNDED; 
  27.     private static final OrderStatus $VALUES[]; 
  28.  
  29.     static  
  30.     { 
  31.         NO_PAY = new OrderStatus("NO_PAY", 0); 
  32.         PAY = new OrderStatus("PAY", 1); 
  33.         REFUNDING = new OrderStatus("REFUNDING", 2); 
  34.         REFUNDED = new OrderStatus("REFUNDED", 3); 
  35.         FAIL_REFUNDED = new OrderStatus("FAIL_REFUNDED", 4); 
  36.         $VALUES = (new OrderStatus[] { 
  37.             NO_PAY, PAY, REFUNDING, REFUNDED, FAIL_REFUNDED 
  38.         }); 
  39.     } 

如源碼所示:

  • 編譯器會自動幫我們創建一個 final 類型的類繼承 Enum 類,所以枚舉類不能被繼承。
  • 會自動生成私有構造方法,當然我們也可以定義構造方法,但必須是私有的,這樣就不能在別處聲明此類的對象了。
  • 枚舉項會被自動添加 public static final 修飾,并定義為 OrderStatus 類型,并在靜態代碼塊中被初始化。
  • 并提供了 values() 和 valueOf(String name) 的靜態方法。

我們定義的枚舉變量實際上是編譯器幫我們自動生成了構造函數。

所有枚舉類都是 Enum 的子類,枚舉類可以實現一個或多個接口。

Enum

Enum 是所有 Java 語言枚舉類型的公共基類,實現了 Comparable 和 Serializable 接口。它包含 final 類型的 name 和 ordinal (此枚舉常量的序號,從0開始)屬性,下面我們來了解下它的方法

  • protected Enum(String name, int ordinal);——構造方法;
  • public String toString();——返回 name 字段,即枚舉定義枚舉變量的字符串;
  • protected final Object clone();——拋出 CloneNotSupportedException 異常,保證枚舉類永遠不會被克隆;
  • public final ClassgetDeclaringClass();——返回與此枚舉常量的枚舉類型對應的類對象;
  • protected final void finalize();—— 枚舉類不能有 finalize 方法;
  • readObject(ObjectInputStream in);& readObjectNoData();—— 拋出InvalidObjectException 異常,防止默認反序列化;

擴展

枚舉類中可以自定義屬性

自定義的屬性值最好用 private final 修飾,防止生成的 set 方法在使用時修改屬性值,使代碼更加安全。

枚舉類中可以自定義構造函數

構造函數必須為 private 修飾,防止在別處聲明此類對象。

枚舉類可以自定義方法,枚舉項可以選擇性覆蓋自定義的方法。

  1. public enum OrderStatus{ 
  2.     NO_PAY("未支付",0), 
  3.     PAY("已支付",1){ 
  4.         @Override 
  5.         public void printOrderStatus() { 
  6.             System.out.println("已支付"); 
  7.         } 
  8.     }, 
  9.     REFUNDING("退款中",2), 
  10.     REFUNDED("退款成功",3), 
  11.     FAIL_REFUNDED("退款失敗",4), 
  12.     ; 
  13.  
  14.     private final String name
  15.     private final int status; 
  16.  
  17.     private OrderStatus(String name,int status){ 
  18.         this.name = name
  19.         this.status = status; 
  20.     } 
  21.  
  22.     public void printOrderStatus(){ 
  23.         System.out.println("打印訂單狀態"); 
  24.     } 
  25.  
  26.  
  27. public class EnumTest { 
  28.     public static void main(String[] args) { 
  29.         OrderStatus.PAY.printOrderStatus(); 
  30.         OrderStatus.NO_PAY.printOrderStatus(); 
  31.     } 

枚舉類也可以有抽象方法,但是枚舉項必須重寫該方法。

枚舉類實現接口

與普通類一樣,實現接口的時候需要實現接口的抽象方法,也可以讓枚舉類的不同對象實現不同的行為。

  1. //定義一個接口 
  2. public interface Order { 
  3.     void printOrderStatus(); 
  4.  
  5. //枚舉類實現該接口 
  6. public enum OrderStatus implements Order
  7.     NO_PAY("未支付",0){ 
  8.         @Override 
  9.         public void printOrderStatus() { 
  10.             System.out.println("未支付"); 
  11.         } 
  12.     }, 
  13.     PAY("已支付",1){ 
  14.         @Override 
  15.         public void printOrderStatus() { 
  16.             System.out.println("已支付"); 
  17.         } 
  18.     }, 
  19.     REFUNDING("退款中",2){ 
  20.         @Override 
  21.         public void printOrderStatus() { 
  22.             System.out.println("退款中"); 
  23.         } 
  24.     }, 
  25.     REFUNDED("退款成功",3){ 
  26.         @Override 
  27.         public void printOrderStatus() { 
  28.             System.out.println("退款成功"); 
  29.         } 
  30.     }, 
  31.     FAIL_REFUNDED("退款失敗",4){ 
  32.         @Override 
  33.         public void printOrderStatus() { 
  34.             System.out.println("退款失敗"); 
  35.         } 
  36.     }, 
  37.     ; 
  38.  
  39.     private final String name
  40.     private final int status; 
  41.  
  42.     private OrderStatus(String name,int status){ 
  43.         this.name = name
  44.         this.status = status; 
  45.     } 

此時查看編譯后的文件,會發現除了生成 OrderStatus.class 文件之外,還生成了多個 .class 文件:

它們是 OrderStatus.class 中生成的匿名內部類的文件。

狀態轉換

需求

訂單是電商項目中不可缺少的組成部分,而訂單狀態的轉換也是我們經常討論的問題。我們都知道訂單狀態的轉換是有一定的邏輯性的,不可以隨意轉換。

例:你想購買某個商品,只是把它加入了購物車,此時應該是未支付狀態。如果來個請求想把它轉換為退款狀態,那么系統應該拋出提示信息“狀態轉換失敗,請先完成購買!”

接下來我們就用枚舉來完成一下訂單狀態轉換的限制。

實現

枚舉類定義:

  1. public enum OrderStatus{ 
  2.     NO_PAY("未支付",0){ 
  3.         @Override 
  4.         public Boolean canChange(OrderStatus orderStatus) { 
  5.             switch (orderStatus){ 
  6.                 case PAY: 
  7.                     return true
  8.                 default
  9.                     return false
  10.             } 
  11.         } 
  12.     }, 
  13.     PAY("已支付",1){ 
  14.         @Override 
  15.         public Boolean canChange(OrderStatus orderStatus) { 
  16.             //因為退款接口一般都會有延遲,所以會先轉化為“退款中”狀態 
  17.             switch (orderStatus){ 
  18.                 case REFUNDING: 
  19.                     return true
  20.                 default
  21.                     return false
  22.             } 
  23.         } 
  24.     }, 
  25.     REFUNDING("退款中",2){ 
  26.         @Override 
  27.         public Boolean canChange(OrderStatus orderStatus) { 
  28.             switch (orderStatus){ 
  29.                 case REFUNDED: 
  30.                 case FAIL_REFUNDED: 
  31.                     return true
  32.                 default
  33.                     return false
  34.             } 
  35.         } 
  36.     }, 
  37.     REFUNDED("退款成功",3), 
  38.     FAIL_REFUNDED("退款失敗",4), 
  39.     ; 
  40.  
  41.     private final String name
  42.     private final int status; 
  43.  
  44.     private OrderStatus(String name,int status){ 
  45.         this.name = name
  46.         this.status = status; 
  47.     } 
  48.  
  49.     //自定義轉換方法 
  50.     public Boolean canChange(OrderStatus orderStatus){ 
  51.         return false
  52.     } 

調用方法:

  1. public class EnumTest { 
  2.  
  3.     public static void main(String[] args) { 
  4.         Boolean aBoolean = OrderStatus.NO_PAY.canChange(OrderStatus.PAY); 
  5.         String statusStr = aBoolean?"可以":"不可以"
  6.         System.out.println("是否可以完成狀態轉換:"+ statusStr); 
  7.  
  8.         Boolean flag = OrderStatus.REFUNDED.canChange(OrderStatus.FAIL_REFUNDED); 
  9.         String flagStr = flag?"可以":"不可以"
  10.         System.out.println("是否可以完成狀態轉換:"+ flagStr); 
  11.     } 

返回結果:

 

這樣我們就用枚舉類實現了訂單狀態轉換的限制。此例子只是為狀態轉換提供一種思路,具體的流程還需要根據自己系統中的業務來具體處理。

 

責任編輯:武曉燕 來源: 阿Q說代碼
相關推薦

2016-06-02 15:10:12

SwiftSelector

2022-02-14 08:04:02

Go語法糖編譯器

2024-09-11 16:34:38

語法糖Java語言

2016-10-14 14:04:34

JAVA語法main

2020-12-08 07:51:53

Java語法糖泛型

2010-01-22 17:55:23

VB.NET語法糖

2024-10-11 15:04:35

KafkaLeader選舉

2022-02-17 13:46:15

SSH命令內網

2019-05-23 11:42:04

Java語法糖編程語言

2024-10-31 15:16:35

2019-06-05 13:05:47

PHP語法糖編碼

2022-08-04 14:38:49

vue3.2setup代碼

2023-09-01 10:00:17

2023-09-11 08:20:17

對象閉包底層

2009-08-18 12:52:33

C#枚舉類型

2025-05-28 08:25:00

JavaScript代碼開發

2012-03-31 16:42:24

2011-08-08 12:44:30

IP安全策略遠程登錄

2018-07-30 16:32:25

應屆生認知誤區

2019-05-14 14:51:40

Java語法糖用法
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产成人免费视频网站高清观看视频 | 亚洲人人 | 午夜影视在线观看 | 日韩在线免费播放 | 美国av片在线观看 | 午夜精品久久久久久久久久久久久 | 中文久久 | 久久亚洲天堂 | 伊人无码高清 | 精品在线一区 | 91精品国产自产在线老师啪 | 欧美日韩精品亚洲 | 免费看91 | 蜜臀91视频 | 黄免费在线 | 国产视频一区二区 | 成人精品一区亚洲午夜久久久 | 免费视频一区 | 91免费电影 | 欧美在线观看一区 | 99re热精品视频国产免费 | 中文字幕 在线观看 | 黑色丝袜三级在线播放 | 亚洲小说图片 | 亚洲区中文字幕 | 四虎伊人 | 国产专区视频 | 日韩精品二区 | 国产精品欧美日韩 | 亚洲午夜精品一区二区三区他趣 | 国产精品美女一区二区 | 欧美一级片中文字幕 | 蜜臀久久 | 国产精品成人久久久久a级 久久蜜桃av一区二区天堂 | 国产亚洲网站 | 综合中文字幕 | 中文字幕av亚洲精品一部二部 | 综合一区二区三区 | 国产精品久久久久久久久久久久 | 免费看片在线播放 | 国产成人精品一区二区三区网站观看 |