淺談 Guava 中的 ImmutableMap.of 方法的坑

作者:明明如月學長, CSDN 博客專家,大廠高級 Java 工程師,《性能優化方法論》作者、《解鎖大廠思維:剖析《阿里巴巴Java開發手冊》》、《再學經典:《EffectiveJava》獨家解析》專欄作者。

熱門文章推薦

  • (1)《為什么很多人工作 3 年 卻只有 1 年經驗?》
  • (2)《從失望到精通:AI 大模型的掌握與運用技巧》
  • (3)《AI 時代,程序員的出路在何方?》
  • (4)《如何寫出高質量的文章:從戰略到戰術》
  • (5)《我的技術學習方法論》
  • (6)《我的性能方法論》
  • (7)《AI 時代的學習方式: 和文檔對話》

一、背景

Guava 的 ImmutableMap類提供了 of方法,可以很方便地構造不可變 Map。

 ImmutableMap<Object, Object> build = ImmutableMap.of("a",1,"b",2);

然而,實際工作開發中很多人會從開始認為非常方便,后面到發現很多大家都會遇到相似的“問題”。
比如 ImmutableMap類的 of 存在很多重載的方法,但是最多只有五個鍵值對。
有無參的方法:

  /*** Returns the empty map. This map behaves and performs comparably to {@link* Collections#emptyMap}, and is preferable mainly for consistency and maintainability of your* code.** <p><b>Performance note:</b> the instance returned is a singleton.*/@SuppressWarnings("unchecked")public static <K, V> ImmutableMap<K, V> of() {return (ImmutableMap<K, V>) RegularImmutableMap.EMPTY;}

有支持一個鍵值對的方法:

  /*** Returns an immutable map containing a single entry. This map behaves and performs comparably to* {@link Collections#singletonMap} but will not accept a null key or value. It is preferable* mainly for consistency and maintainability of your code.*/public static <K, V> ImmutableMap<K, V> of(K k1, V v1) {return ImmutableBiMap.of(k1, v1);}

到支持五個鍵值對的方法:

  /*** Returns an immutable map containing the given entries, in order.** @throws IllegalArgumentException if duplicate keys are provided*/public static <K, V> ImmutableMap<K, V> of(K k1, V v1, K k2, V v2, K k3, V v3, K k4, V v4, K k5, V v5) {return RegularImmutableMap.fromEntries(entryOf(k1, v1), entryOf(k2, v2), entryOf(k3, v3), entryOf(k4, v4), entryOf(k5, v5));}

很多人會遇到的坑:

  • 超過五個鍵值對怎么辦?
  • key 和 value “居然”都不能為 null?
  • 同一個 key 重復 put 報錯

二、場景還原

2.1 超過 5 個鍵值對問題

雖然 of 方法很好用,但是經常會遇到超過 5 個鍵值對的情況,就非常不方便。

解法1:升級版本

在 guava 31.0 版本以后,已經拓展到了 10 個鍵值對!

  /*** Returns an immutable map containing the given entries, in order.** @throws IllegalArgumentException if duplicate keys are provided* @since 31.0*/public static <K, V> ImmutableMap<K, V> of(K k1,V v1,K k2,V v2,K k3,V v3,K k4,V v4,K k5,V v5,K k6,V v6,K k7,V v7,K k8,V v8,K k9,V v9,K k10,V v10) {return RegularImmutableMap.fromEntries(entryOf(k1, v1),entryOf(k2, v2),entryOf(k3, v3),entryOf(k4, v4),entryOf(k5, v5),entryOf(k6, v6),entryOf(k7, v7),entryOf(k8, v8),entryOf(k9, v9),entryOf(k10, v10));}

解法2:使用 builder 方法

com.google.common.collect.ImmutableMap#builder 方法可以通過構造器的方式不斷 put 鍵值對,最后 build即可,也非常方便。

      ImmutableMap<Object, Object> build = ImmutableMap.builder().put("a", 1).put("b", 2).put("c", 3).put("d",4).put("e",5).put("f",6).build();

也可以參考 2.2 中的解法。

2.2 鍵值都不允許為 null

復現

很多人看到名字就知道不可“修改” 但不太清楚它的鍵值都不允許為 null。

key 為空的情況:
image.png

value 為空的情況:
image.png

真正開發時不會那么簡單,有時候需要調用某個接口獲取返回值然后再構造一個不可編輯的 Map 返回給下游使用。很可能在測試的時候都沒有出現 null 值,發布上線,發現 key 或者 value 為 null,就會造成線上問題 或者 bug。

源碼

對于 of的多參數重載:

  /*** Returns an immutable map containing the given entries, in order.** @throws IllegalArgumentException if duplicate keys are provided*/public static <K, V> ImmutableMap<K, V> of(K k1, V v1, K k2, V v2, K k3, V v3) {return RegularImmutableMap.fromEntries(entryOf(k1, v1), entryOf(k2, v2), entryOf(k3, v3));}
  /*** Verifies that {@code key} and {@code value} are non-null, and returns a new immutable entry* with those values.** <p>A call to {@link Entry#setValue} on the returned entry will always throw {@link* UnsupportedOperationException}.*/static <K, V> Entry<K, V> entryOf(K key, V value) {return new ImmutableMapEntry<>(key, value);}
  ImmutableMapEntry(K key, V value) {super(key, value);checkEntryNotNull(key, value);}
  static void checkEntryNotNull(Object key, Object value) {if (key == null) {throw new NullPointerException("null key in entry: null=" + value);} else if (value == null) {throw new NullPointerException("null value in entry: " + key + "=null");}}

當然,如果你比較心細的話會發現 IDE 中會有警告,也可以很大程度上避免這個問題。

解法

不如換個“殊途同歸”的辦法,先用 HashMap 去實現同一個 key 的值覆蓋的功能,然后通過 Collections.unmodifiableMap來實現不可編輯功能。

     Map<String, Object> map = new HashMap<>();map.put("a", 1);map.put("b", 2);map.put("c", 3);map.put("d", 4);map.put("e", 5);map.put("f", null);Map<String, Object> unmodifiableMap = Collections.unmodifiableMap(map);System.out.println(unmodifiableMap);

在這里插入圖片描述

2.3 key 重復報錯

復現

如果一不小心 key 重復,也會報 java.lang.IllegalArgumentException異常。

        ImmutableMap<Object, Object> build = ImmutableMap.builder().put("a", 1).put("b", 2).put("c", 3).put("d",4).put("f",5).put("f",6).build();System.out.println(build);

image.png

源碼

  /*** Returns an immutable map containing the given entries, in order.** @throws IllegalArgumentException if duplicate keys are provided*/public static <K, V> ImmutableMap<K, V> of(K k1, V v1, K k2, V v2) {return RegularImmutableMap.fromEntries(entryOf(k1, v1), entryOf(k2, v2));}

最底層會對 entry 進行校驗:

  /*** Checks if the given key already appears in the hash chain starting at {@code keyBucketHead}. If* it does not, then null is returned. If it does, then if {@code throwIfDuplicateKeys} is true an* {@code IllegalArgumentException} is thrown, and otherwise the existing {@link Entry} is* returned.** @throws IllegalArgumentException if another entry in the bucket has the same key and {@code*     throwIfDuplicateKeys} is true* @throws BucketOverflowException if this bucket has too many entries, which may indicate a hash*     flooding attack*/@CanIgnoreReturnValuestatic <K, V> @Nullable ImmutableMapEntry<K, V> checkNoConflictInKeyBucket(Object key,Object newValue,@CheckForNull ImmutableMapEntry<K, V> keyBucketHead,boolean throwIfDuplicateKeys)throws BucketOverflowException {int bucketSize = 0;for (; keyBucketHead != null; keyBucketHead = keyBucketHead.getNextInKeyBucket()) {if (keyBucketHead.getKey().equals(key)) {if (throwIfDuplicateKeys) {checkNoConflict(/* safe= */ false, "key", keyBucketHead, key + "=" + newValue);} else {return keyBucketHead;}}if (++bucketSize > MAX_HASH_BUCKET_LENGTH) {throw new BucketOverflowException();}}return null;}

最終報錯:

  static IllegalArgumentException conflictException(String conflictDescription, Object entry1, Object entry2) {return new IllegalArgumentException("Multiple entries with same " + conflictDescription + ": " + entry1 + " and " + entry2);}

解法

ImmutableMapbuilder除了提供 buid 之外, 在 31.0 版本之后還通過了 buildKeepingLastbuildOrThrow
image.png
可以通過 buildKeepingLast設置當 key 重復時取后面的值。

    /*** Returns a newly-created immutable map. The iteration order of the returned map is the order* in which entries were inserted into the builder, unless {@link #orderEntriesByValue} was* called, in which case entries are sorted by value.** <p>Prefer the equivalent method {@link #buildOrThrow()} to make it explicit that the method* will throw an exception if there are duplicate keys. The {@code build()} method will soon be* deprecated.** @throws IllegalArgumentException if duplicate keys were added*/public ImmutableMap<K, V> build() {return buildOrThrow();}/*** Returns a newly-created immutable map, or throws an exception if any key was added more than* once. The iteration order of the returned map is the order in which entries were inserted* into the builder, unless {@link #orderEntriesByValue} was called, in which case entries are* sorted by value.** @throws IllegalArgumentException if duplicate keys were added* @since 31.0*/public ImmutableMap<K, V> buildOrThrow() {return build(true);}/*** Returns a newly-created immutable map, using the last value for any key that was added more* than once. The iteration order of the returned map is the order in which entries were* inserted into the builder, unless {@link #orderEntriesByValue} was called, in which case* entries are sorted by value. If a key was added more than once, it appears in iteration order* based on the first time it was added, again unless {@link #orderEntriesByValue} was called.** <p>In the current implementation, all values associated with a given key are stored in the* {@code Builder} object, even though only one of them will be used in the built map. If there* can be many repeated keys, it may be more space-efficient to use a {@link* java.util.LinkedHashMap LinkedHashMap} and {@link ImmutableMap#copyOf(Map)} rather than* {@code ImmutableMap.Builder}.** @since 31.1*/public ImmutableMap<K, V> buildKeepingLast() {return build(false);}

低版本的話可以考慮先用 HashMap 構造數據,然后使用 com.google.common.collect.ImmutableMap#copyOf(java.util.Map<? extends K,? extends V>) 轉換即可。

    Map<String, Object> map = new HashMap<>();map.put("a", 1);map.put("b", 2);map.put("c", 3);map.put("d", 4);map.put("f", 5);map.put("f", 6);ImmutableMap<Object, Object> build = ImmutableMap.copyOf(map);System.out.println(build);

三、為什么?

3.1 為什么默認是 5 個鍵值對?

其實 31.0 版本,已經支持 10 個鍵值對了。
此處,斗膽猜測,of方法僅是為了提供更簡單的構造 ImmutableMap的方法,而“通常” 5 個就足夠了。
然而,實踐中很多人發現 5 個并不夠,因此高版本中支持 10個鍵值對。

Guava 也有相關 Issues 的討論 ImmutableMap::of should accept more entries #2071
https://github.com/google/guava/issues/2071

image.png

3.2 為什么不允許鍵值為 null ?

Github 上也有相關討論:
Question: Why RegularImmutableMap.fromEntryArray enforces “not null” policy on values? #5844

image.png

wiki 上有相關解釋:
https://github.com/google/guava/wiki/UsingAndAvoidingNullExplained

使用 ChatGPT 對上述 wiki 進行關鍵信息提取:

在谷歌的 Guava 庫的設計哲學中,不允許在 ImmutableMap(或其他類似的集合)中使用 null 值有幾個關鍵原因:

防止錯誤:Guava 團隊發現在 Google 的代碼庫中,大約 95% 的集合不應包含任何 null 值。允許 null 值會增加出錯的風險,比如可能導致空指針異常。讓這些集合在遇到 null 時快速失敗(fail-fast)而不是默默接受 null,對開發者來說更有幫助。

消除歧義:null 值的含義通常不明確。例如,在使用 Map.get(key) 時,如果返回 null,可能是因為映射中該鍵對應的值為 null,或者該鍵在映射中不存在。這種歧義會導致理解和使用上的困難。

提倡更清晰的實踐:在 Set 或 Map 中使用 null 值通常不是一個好的做法。更清晰的方法是在查找操作中顯式處理 null,例如,如果你想在 Map 中使用 null 作為值,最好將那個條目留空,并保持一個單獨的非空鍵集合。這樣做可以避免混淆那些映射中鍵存在但值為 null,和那些映射中根本沒有該鍵的情況。

選擇適當的替代方案:如果你確實需要使用 null 值,并且遇到了不友好處理 null 的集合實現時,Guava 建議使用不同的實現。例如,如果 ImmutableList 不滿足需求,可以使用 Collections.unmodifiableList(Lists.newArrayList()) 作為替代。

總體而言,Guava 庫通過避免在其集合中使用 null,旨在提供更清晰、更健壯、且更易于維護的代碼實踐。

3.3 為什么重復 key 會報錯?

我認為,主要是為了符合“不可變”的語義,既然是不可變,那么相同的 key 不應該重復放入到 map 中。其次,也可以避免意外的數據覆蓋或丟失。

四、總結

雖然這個問題并不難,但很多人并不知道會有那么多“坑”,很多人都需要重復思考如何解決這些限制。
因此,本文總結在這里,希望對大家有幫助。


在這里插入圖片描述

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/165372.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/165372.shtml
英文地址,請注明出處:http://en.pswp.cn/news/165372.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

vue項目下.env.development環境變量配置文件

.env.development 文件是一個用于開發環境配置的文件。在許多應用程序中&#xff0c;開發環境和生產環境具有不同的配置需求。.env.development 文件允許你在開發環境中定義特定的環境變量和配置選項。 一般來說&#xff0c;.env.development 文件用于存儲開發環境相關的配置信…

國自然項目基金撰寫的隱藏技巧、范例分析及提交前的自我審查

目錄 一、基金項目申請要求、重點及項目介紹 二、基金的撰寫技巧 三、基金撰寫的隱藏技巧 四、范例分析及提交前的自我審查 更多應用 基金項目申請需要進行跨學科的技術融合&#xff0c;申請人需要與不同領域結合&#xff0c;形成多學科交叉的研究。基金項目申請在新時期更…

由紅黑樹引出的HashMap擴容機制的思考

紅黑樹是什么&#xff1f; 三大特點&#xff1a; 根節點是黑色&#xff0c;葉節點是不存儲數據的黑色空節點 任何相鄰的兩個節點不能同時為紅色 任意節點到其可到達的節點間包含相同數量的黑色節點 聯想&#xff1a;Java HashMap底層紅黑樹原理 HashMap基于哈希表Map接口實…

快速掌握Pyqt5的三種主窗口

PyQt5是一個強大的跨平臺GUI框架&#xff0c;它提供了多種不同類型的主窗口類&#xff0c;以滿足不同的應用需求。下面是PyQt5中最常見的幾種主窗口類型及其創建方式的簡介&#xff1a; 1. QMainWindow QMainWindow是用于創建具有菜單欄、工具欄、狀態欄和中心窗口部件&#…

內存池 示例一

內存池是一種管理內存分配和釋放的技術&#xff0c;用于優化內存的使用效率。它通過預先分配一塊內存區域&#xff0c;并將其劃分為多個較小的塊&#xff08;內存塊池&#xff09;&#xff0c;然后按需分配這些內存塊來減少內存碎片化和頻繁的系統調用。這些內存塊可以是相同大…

Centos7.9配置nfs共享及rsync同步

客戶需求對oracle數據庫做一個跨機房的備份&#xff0c;原環境已做rman備份和每天expdp全庫導出&#xff0c;遠端只有虛擬化環境&#xff0c;可提供一個虛擬機&#xff0c;2個機房間網絡互通。 首先配置nfs服務端 查看操作系統版本 [rootnas199 ~]# more /etc/redhat-relea…

Python面經【1】

一、協程的相關概念 協程&#xff08;又稱微線程&#xff09;運行在線程之上&#xff0c;更加輕量級&#xff0c;協程并沒有增加線程總數&#xff0c;只是在線程的基礎上通過分時復用的方式運行多個協程&#xff0c;大大提高工程效率。 協程的特點&#xff1a; 輕量級&#…

WordPress站點屏蔽過濾垃圾評論教程(Akismet反垃圾評論插件)

前段時間我的WordPress站點經常收到垃圾評論的轟炸&#xff0c;嚴重時一天會收到幾十條垃圾評論。我這個小破站一沒啥流量&#xff0c;二又不盈利&#xff0c;實在是不太理解為啥有人要這么執著地浪費資源在上面。 Akismet反垃圾評論插件 其實用了 Akismet 反垃圾評論插件后&a…

快速掌握Pyqt5的6種按鈕

在PyQt5中&#xff0c;按鈕是構建用戶界面的基本元素之一&#xff0c;用于執行命令、啟動功能或觸發事件。PyQt5提供了多種類型的按鈕&#xff0c;每種都適用于不同的場景和需求。 1. QPushButton QPushButton 是最常用的按鈕類型&#xff0c;適用于大多數情況&#xff0c;如…

ARCore:在Android上構建令人驚嘆的增強現實體驗

ARCore&#xff1a;在Android上構建令人驚嘆的增強現實體驗 一、 AR 介紹1.1 AR技術簡介1.2 AR技術原理1.3 AR技術應用領域 二、Google的增強現實平臺ARCore2.1 ARCore簡介2.2 ARCore API介紹2.3 ARCore API使用示例 三、總結 一、 AR 介紹 增強現實 Augmented Reality&#x…

【算法-字符串2】替換空格 + 反轉單詞

今天&#xff0c;帶來字符串相關算法的講解。文中不足錯漏之處望請斧正&#xff01; 理論基礎點這里 1. 替換空格 題目描述&#xff1a;請實現一個函數&#xff0c;把字符串 s 中的每個空格替換成"%20"。 來源&#xff1a;力扣&#xff08;LeetCode&#xff09; 難…

Lettuce使用詳解

簡介特點連接池連接池特點連接池管理連接池優勢連接池配置參數 監控常用監控工具通過JMX監控通過Prometheus監控 代碼示例拓展springboot中通過jmx上報到Prometheus代碼示例更多Redis相關內容 簡介 Lettuce 是一個高級的、線程安全的 Redis 客戶端&#xff0c;用于與 Redis 數…

深度學習基礎概念

1. 神經網絡基礎 神經元&#xff08;Neuron&#xff09;&#xff1a; 了解神經網絡的基本組成單元。激活函數&#xff08;Activation Function&#xff09;&#xff1a; 學習常見的激活函數&#xff0c;如Sigmoid、ReLU等&#xff0c;以及它們在神經網絡中的作用。前饋神經網絡…

An issue was found when checking AAR metadata

一、報錯信息 An issue was found when checking AAR metadata:1. Dependency androidx.activity:activity:1.8.0 requires libraries and applications that depend on it to compile against version 34 or later of the Android APIs.:app is currently compiled against …

Python 異步套接字編程

異步套接字編程是異步編程在網絡通信中的應用&#xff0c;它使用異步 IO 操作和事件循環來實現高并發的網絡應用。Python 中的 asyncio 模塊提供了對異步套接字編程的支持&#xff0c;以下是異步套接字編程的一些重要概念和使用方法&#xff1a; 1. 異步套接字服務器&#xff…

git與ssh多賬戶共存

git與ssh多賬戶共存 前言git多賬戶ssh多公鑰參考 前言 在使用git與ssh時&#xff0c;經常會遇到多個賬戶共存的情況 例如使用不同的公鑰登陸到不同的服務&#xff1b;使用不同的git信息進行commit git多賬戶 在默認情況下 git的信息存在 ~/.gitconfig 可以使用命令查看 git…

關于elementui和ant design vue無法禁止瀏覽器自動填充問題

以and design vue 為例&#xff1a; 圖標用來顯隱賬號密碼 html&#xff1a; <a-form-model-item label"賬號密碼:" prop"password"><a-input v-if"passwordTab" ref"passwordInput" v-model"form.password" typ…

詳解最長公共子序列問題(三種方法)

這里&#xff0c;為了更方便地解釋&#xff0c;我以洛谷上的一道典型題目為例&#xff0c;為大家講解處理最長公共子序列問題的幾種常見方法。這道題目中規定了兩個子序列的長度相等&#xff0c;如果遇到不等的情況&#xff0c;也只需要對長度稍作修改即可&#xff0c;算法思想…

qs-一個序列化和反序列化的JavaScript庫

起因 一個業務場景中&#xff0c;最終得到一串字符"status[0]value1&status[1]value2" 通過解析&#xff0c;理應得到一個數組&#xff0c;卻得到一個對象 于是展開問題排查 最終發現是qs.parse 這個地方出了問題 排查結果 qs解析這種帶下標的字符串時&#xff…

基于python的NBA球員數據可視化分析的設計與實現

完整下載&#xff1a;基于python的NBA球員數據可視化分析的設計與實現.docx 基于python的NBA球員數據可視化分析的設計與實現 Design and Implementation of NBA Player Data Visualization Analysis based on Python 目錄 目錄 2 摘要 3 關鍵詞 4 第一章 引言 4 1.1 研究背景 …