Java 列舉查詢並不拋異常的實現
本文由碼農網 – 小峰原創翻譯,轉載請看清文末的轉載要求,歡迎參與我們的付費投稿計劃!
Java Enum是一個非常有用的功能,但很多人通常並不能充分利用,這是因為一些庫不會優先擇用該功能。通常我們也可以正確使用Java列舉功能,但在許多程式碼庫中往往存在著這樣一個問題,於是有了這篇文章。問題很簡單:我們應該如何通過名稱或值獲取列舉,並忽略不存在的值?
列舉
這是我們將在示例中使用的列舉。 挑選更復雜的列舉是為了也可以通過其他欄位展示查詢列舉。
public enum CardColor { RED, BLACK, ; } // Jackson annotation to print the enum as an Object instead of the default name. @JsonFormat(shape = JsonFormat.Shape.OBJECT) public enum CardSuit { // Unicode suits - https://en.wikipedia.org/wiki/Playing_cards_in_Unicode SPADE("Spade", String.valueOf((char) 0x2660), CardColor.BLACK), HEART("Heart", String.valueOf((char) 0x2665), CardColor.RED), DIAMOND("Diamond", String.valueOf((char) 0x2666), CardColor.RED), CLUB("Club", String.valueOf((char) 0x2663), CardColor.BLACK), ; private String displayName; private String symbol; private CardColor color; private CardSuit(String displayName, String symbol, CardColor color) { this.displayName = displayName; this.symbol = symbol; this.color = color; } public String getDisplayName() { return displayName; } public void setDisplayName(String displayName) { this.displayName = displayName; } public String getSymbol() { return symbol; } public void setSymbol(String symbol) { this.symbol = symbol; } public CardColor getColor() { return color; } public void setColor(CardColor color) { this.color = color; }
在 GitHub 上檢視。
問題
在你知道輸入有效的時候,使用Enum.valueOf非常棒。但是,如果傳入無效的名稱,那麼將丟擲異常。在某些情況下,這很好。不過,在通常情況下,我們寧願忽略異常並返回null。
log.debug("Running valueOf"); for (String name : names) { try { log.debug("looking up {} found {}", name, Json.serializer().toString(CardSuit.valueOf(name))); } catch (Exception ex) { log.warn("Exception Thrown", ex); } }
在 GitHub 上檢視。
2017-02-22 14:46:38.556 [main] DEBUG c.s.examples.common.EnumLookup - Running valueOf 2017-02-22 14:46:38.804 [main] DEBUG c.s.examples.common.EnumLookup - looking up SPADE found {"displayName":"Spade","symbol":"♠","color":"BLACK"} 2017-02-22 14:46:38.806 [main] DEBUG c.s.examples.common.EnumLookup - looking up HEART found {"displayName":"Heart","symbol":"♥","color":"RED"} 2017-02-22 14:46:38.806 [main] DEBUG c.s.examples.common.EnumLookup - looking up DIAMOND found {"displayName":"Diamond","symbol":"♦","color":"RED"} 2017-02-22 14:46:38.806 [main] DEBUG c.s.examples.common.EnumLookup - looking up CLUB found {"displayName":"Club","symbol":"♣","color":"BLACK"} 2017-02-22 14:46:38.808 [main] WARN c.s.examples.common.EnumLookup - Exception Thrown java.lang.IllegalArgumentException: No enum constant com.stubbornjava.examples.common.EnumLookup.CardSuit.Missing at java.lang.Enum.valueOf(Enum.java:238) at com.stubbornjava.examples.common.EnumLookup$CardSuit.valueOf(EnumLookup.java:1) at com.stubbornjava.examples.common.EnumLookup.main(EnumLookup.java:154)
拙劣的實現
很不幸的是,以下兩種方法在程式碼庫中出現得是這麼的頻繁。反面例子,勿學。
Enum.valueOf With Try Catch(劣)
這種拙劣的做法最常見於初學者。異常不應該用於控制流,並且這樣可能會有一些效能影響。不要偷懶。你必須用正確的方式去做。
/* * Please don't do this! Using try / catch for * control flow is a bad practice. */ public static CardSuit trycatchValueOf(String name) { try { return CardSuit.valueOf(name); } catch (Exception ex) { log.warn("Exception Thrown", ex); return null; } }
在 GitHub 上檢視。
log.debug("Running trycatchValueOf"); for (String name : names) { log.debug("looking up {} found {}", name, Json.serializer().toString(CardSuit.trycatchValueOf(name))); }
在 GitHub 上檢視。
2017-02-22 14:46:38.809 [main] DEBUG c.s.examples.common.EnumLookup - Running trycatchValueOf 2017-02-22 14:46:38.809 [main] DEBUG c.s.examples.common.EnumLookup - looking up SPADE found {"displayName":"Spade","symbol":"♠","color":"BLACK"} 2017-02-22 14:46:38.809 [main] DEBUG c.s.examples.common.EnumLookup - looking up HEART found {"displayName":"Heart","symbol":"♥","color":"RED"} 2017-02-22 14:46:38.809 [main] DEBUG c.s.examples.common.EnumLookup - looking up DIAMOND found {"displayName":"Diamond","symbol":"♦","color":"RED"} 2017-02-22 14:46:38.809 [main] DEBUG c.s.examples.common.EnumLookup - looking up CLUB found {"displayName":"Club","symbol":"♣","color":"BLACK"} 2017-02-22 14:46:38.809 [main] WARN c.s.examples.common.EnumLookup - Exception Thrown java.lang.IllegalArgumentException: No enum constant com.stubbornjava.examples.common.EnumLookup.CardSuit.Missing at java.lang.Enum.valueOf(Enum.java:238) at com.stubbornjava.examples.common.EnumLookup$CardSuit.valueOf(EnumLookup.java:1) at com.stubbornjava.examples.common.EnumLookup$CardSuit.trycatchValueOf(EnumLookup.java:89) at com.stubbornjava.examples.common.EnumLookup.main(EnumLookup.java:171) 2017-02-22 14:46:38.809 [main] DEBUG c.s.examples.common.EnumLookup - looking up Missing found null
通過迭代查詢(劣)
這種方法也很常見(見這裡),但至少程式設計師知道不能用try/catch來捕獲異常。那麼,這種方法有什麼問題呢?沒錯,它會迭代所有列舉,直到找到匹配的列舉或返回null——最壞的情況下需要n次,其中n就是列舉值的數量。有些人可能會認為這微不足道,這是過早優化了而已。但是,資料結構和演算法是CS基礎。使用Map而不是迭代集合要省力得多。這會大大提高效能嗎?不,但它是一個很好的習慣。在面試候選人時,你會對線性複雜度搜尋演算法感到滿意嗎?此時,你不應該讓這樣的程式碼審查通過。
/* * Please don't do this! It is inefficient and it's * not very hard to use Guava or a static Map as an index. */ public static CardSuit iterationFindByName(String name) { for (CardSuit suit : CardSuit.values()) { if (name.equals(suit.name())) { return suit; } } return null; }
在 GitHub 上檢視。
log.debug("Running iteration"); for (String name : names) { log.debug("looking up {} found {}", name, Json.serializer().toString(CardSuit.iterationFindByName(name))); }
在 GitHub 上檢視。
2017-02-22 14:46:38.808 [main] DEBUG c.s.examples.common.EnumLookup - Running iteration 2017-02-22 14:46:38.809 [main] DEBUG c.s.examples.common.EnumLookup - looking up SPADE found {"displayName":"Spade","symbol":"♠","color":"BLACK"} 2017-02-22 14:46:38.809 [main] DEBUG c.s.examples.common.EnumLookup - looking up HEART found {"displayName":"Heart","symbol":"♥","color":"RED"} 2017-02-22 14:46:38.809 [main] DEBUG c.s.examples.common.EnumLookup - looking up DIAMOND found {"displayName":"Diamond","symbol":"♦","color":"RED"} 2017-02-22 14:46:38.809 [main] DEBUG c.s.examples.common.EnumLookup - looking up CLUB found {"displayName":"Club","symbol":"♣","color":"BLACK"} 2017-02-22 14:46:38.809 [main] DEBUG c.s.examples.common.EnumLookup - looking up Missing found null
較優的實現
以下都可以通過使用Map形式的索引起作用。但是,它們之間也有一些細微的差別。
靜態Map索引(較優)
用於快速查詢固定大小的正確資料結構是什麼?那就是HashMap。現在通過一些額外的樣板,只要我們有很好的雜湊函式的話,我們就可以進行更有效的查詢。稍微更冗長,但如果有辦法減少樣板的話,這將是極好的。
private static final Map<String, CardSuit> nameIndex = Maps.newHashMapWithExpectedSize(CardSuit.values().length); static { for (CardSuit suit : CardSuit.values()) { nameIndex.put(suit.name(), suit); } } public static CardSuit lookupByName(String name) { return nameIndex.get(name); }
在 GitHub 上檢視。
log.debug("Running lookupByName"); for (String name : names) { log.debug("looking up {} found {}", name, Json.serializer().toString(CardSuit.lookupByName(name))); }
在 GitHub 上檢視。
2017-02-22 14:46:38.809 [main] DEBUG c.s.examples.common.EnumLookup - Running lookupByName 2017-02-22 14:46:38.809 [main] DEBUG c.s.examples.common.EnumLookup - looking up SPADE found {"displayName":"Spade","symbol":"♠","color":"BLACK"} 2017-02-22 14:46:38.810 [main] DEBUG c.s.examples.common.EnumLookup - looking up HEART found {"displayName":"Heart","symbol":"♥","color":"RED"} 2017-02-22 14:46:38.810 [main] DEBUG c.s.examples.common.EnumLookup - looking up DIAMOND found {"displayName":"Diamond","symbol":"♦","color":"RED"} 2017-02-22 14:46:38.813 [main] DEBUG c.s.examples.common.EnumLookup - looking up CLUB found {"displayName":"Club","symbol":"♣","color":"BLACK"} 2017-02-22 14:46:38.813 [main] DEBUG c.s.examples.common.EnumLookup - looking up Missing found null
Guava Enums.getIfPresent(推薦)
這是一個常見的用例,我們在Google的朋友為此提供了一個非常乾淨且無需樣板的解決方案。探其究竟,它甚至使用了WeakReferences和WeakHashMaps。基本說來,這段程式碼將建立一個全域性的靜態地圖,並以Enum的類名稱鍵入,並將其用於查詢。
public static CardSuit getIfPresent(String name) { return Enums.getIfPresent(CardSuit.class, name).orNull(); }
在 GitHub 上檢視。
log.debug("Running Guava getIfPresent"); for (String name : names) { log.debug("looking up {} found {}", name, Json.serializer().toString(CardSuit.getIfPresent(name))); }
在 GitHub 上檢視。
2017-02-22 14:46:38.813 [main] DEBUG c.s.examples.common.EnumLookup - Running Guava getIfPresent 2017-02-22 14:46:38.814 [main] DEBUG c.s.examples.common.EnumLookup - looking up SPADE found {"displayName":"Spade","symbol":"♠","color":"BLACK"} 2017-02-22 14:46:38.814 [main] DEBUG c.s.examples.common.EnumLookup - looking up HEART found {"displayName":"Heart","symbol":"♥","color":"RED"} 2017-02-22 14:46:38.815 [main] DEBUG c.s.examples.common.EnumLookup - looking up DIAMOND found {"displayName":"Diamond","symbol":"♦","color":"RED"} 2017-02-22 14:46:38.815 [main] DEBUG c.s.examples.common.EnumLookup - looking up CLUB found {"displayName":"Club","symbol":"♣","color":"BLACK"} 2017-02-22 14:46:38.815 [main] DEBUG c.s.examples.common.EnumLookup - looking up Missing found null
通過欄位的進一步索引
這個完全相同的方法可以用於列舉的其他欄位。想要通過其顯示的名稱或其他屬性來查詢列舉並不少見。
通過欄位索引的靜態Map(較優)
與上述相同的方法,但是在顯示名稱上而不是列舉名稱上進行索引。
private static final Map<String, CardSuit> displayNameIndex = Maps.newHashMapWithExpectedSize(CardSuit.values().length); static { for (CardSuit suit : CardSuit.values()) { displayNameIndex.put(suit.getDisplayName(), suit); } } public static CardSuit lookupByDisplayName(String name) { return displayNameIndex.get(name); }
在 GitHub 上檢視。
log.debug("Running lookupByDisplayName"); for (String displayName : displayNames) { log.debug("looking up {} found {}", displayName, Json.serializer().toString(CardSuit.lookupByDisplayName(displayName))); }
在 GitHub 上檢視。
2017-02-22 14:46:38.815 [main] DEBUG c.s.examples.common.EnumLookup - Running lookupByDisplayName 2017-02-22 14:46:38.815 [main] DEBUG c.s.examples.common.EnumLookup - looking up Spade found {"displayName":"Spade","symbol":"♠","color":"BLACK"} 2017-02-22 14:46:38.815 [main] DEBUG c.s.examples.common.EnumLookup - looking up Heart found {"displayName":"Heart","symbol":"♥","color":"RED"} 2017-02-22 14:46:38.815 [main] DEBUG c.s.examples.common.EnumLookup - looking up Diamond found {"displayName":"Diamond","symbol":"♦","color":"RED"} 2017-02-22 14:46:38.816 [main] DEBUG c.s.examples.common.EnumLookup - looking up Club found {"displayName":"Club","symbol":"♣","color":"BLACK"} 2017-02-22 14:46:38.816 [main] DEBUG c.s.examples.common.EnumLookup - looking up Missing found null
通過欄位索引的靜態Map(較優)
我們不能在這裡利用Guava,因為對於靜態索引建立唯一的全域性金鑰將是一件困難的事。但是,這並不意味著我們沒有幫手!
public class EnumUtils { public static <T, E extends Enum<E>> Function<T, E> lookupMap(Class<E> clazz, Function<E, T> mapper) { @SuppressWarnings("unchecked") E[] emptyArray = (E[]) Array.newInstance(clazz, 0); return lookupMap(EnumSet.allOf(clazz).toArray(emptyArray), mapper); } public static <T, E extends Enum<E>> Function<T, E> lookupMap(E[] values, Function<E, T> mapper) { Map<T, E> index = Maps.newHashMapWithExpectedSize(values.length); for (E value : values) { index.put(mapper.apply(value), value); } return (T key) -> index.get(key); } }
在 GitHub 上檢視。
現在我們有了一個相與樣板沒多大關聯的通用解決方案。
private static final Function<String, CardSuit> func = EnumUtils.lookupMap(CardSuit.class, e -> e.getDisplayName()); public static CardSuit lookupByDisplayNameUtil(String name) { return func.apply(name); }
在 GitHub 上檢視。
log.debug("Running lookupByDisplayNameUtil"); for (String displayName : displayNames) { log.debug("looking up {} found {}", displayName, Json.serializer().toString(CardSuit.lookupByDisplayNameUtil(displayName))); }
在 GitHub 上檢視。
2017-02-22 14:46:38.816 [main] DEBUG c.s.examples.common.EnumLookup - Running lookupByDisplayNameUtil 2017-02-22 14:46:38.816 [main] DEBUG c.s.examples.common.EnumLookup - looking up Spade found {"displayName":"Spade","symbol":"♠","color":"BLACK"} 2017-02-22 14:46:38.816 [main] DEBUG c.s.examples.common.EnumLookup - looking up Heart found {"displayName":"Heart","symbol":"♥","color":"RED"} 2017-02-22 14:46:38.816 [main] DEBUG c.s.examples.common.EnumLookup - looking up Diamond found {"displayName":"Diamond","symbol":"♦","color":"RED"} 2017-02-22 14:46:38.816 [main] DEBUG c.s.examples.common.EnumLookup - looking up Club found {"displayName":"Club","symbol":"♣","color":"BLACK"} 2017-02-22 14:46:38.816 [main] DEBUG c.s.examples.common.EnumLookup - looking up Missing found null
結論
這裡有若干方法都可用於解決同一個問題。有些拙劣,有些更優。
譯文連結:http://www.codeceo.com/article/java-enum-lookup-no-exception.html
英文原文:Java Enum Lookup by Name or Field Without Throwing Exceptions
翻譯作者:碼農網 – 小峰
[ 轉載必須在正文中標註並保留原文連結、譯文連結和譯者等資訊。]
相關文章
- Java列舉-通過值查詢對應的列舉Java
- hibernate異常之--count查詢異常
- Mybatis實現條件IN查詢(foreach)和invalid comparison異常MyBatis
- Java enum列舉類詳解 列舉的常見用法Java
- Java中的Exception拋異常對效能的影響 - BaeldungJavaException
- 並查集java實現並查集Java
- 並查集-Java實現並查集Java
- 恕我直言,我懷疑你並不會用 Java 列舉Java
- Java常出現的異常解決方法總結(不斷更新)Java
- 面試:Java 實現查詢旋轉陣列的最小數字面試Java陣列
- Java 利用列舉實現單例模式Java單例模式
- 深入淺出 Java 中列舉的實現原理Java
- 125 列舉實現PHP擷取中文不亂碼的實現方法PHP
- 日常Bug排查-拋異常不回滾
- 一次分割槽查詢異常的分析
- 執行ArrayList的remove(object)方法拋異常?REMObject
- JavaScript實現常見查詢演算法JavaScript演算法
- 解決Java執行過程中拋簽名異常的問題Java
- 異常-自定義異常的實現和測試
- Java 列舉(enum) 詳解7種常見的用法Java
- c++11 實現列舉值到列舉名的轉換C++
- Java捕獲非檢查異常----UncaughtExceptionHandler的使用JavaException
- MySQL 並列排名和順序排名查詢MySql
- java中的列舉Java
- Java 列舉、JPA 和 PostgreSQL 列舉JavaSQL
- Java列舉Java
- 633. 平方數之和 ( 列舉 + 二分查詢 )
- 並查集(一)並查集的幾種實現並查集
- JVM系列(四):java方法的查詢過程實現JVMJava
- Java實現遞迴查詢樹結構Java遞迴
- 詭異的”慢查詢“
- SSH:hiberate實現資料的查詢(單查詢和全查詢)
- .net core 拋異常對效能影響的求證之路
- Java 列舉 switch的用法Java
- java中的異常Java
- Java 中的異常Java
- 異常中的異常——藉助系統異常處理特例實現匪夷所思的漏洞利用
- Java 列舉(enum)Java
- java列舉類Java