Guava輕松創(chuàng)建和管理不可變集合方法技巧
第1章:引言
大家好,我是小黑。今天,我們來聊聊一個(gè)在Java編程里超有用的話題:使用Guava創(chuàng)建和管理不可變集合。首先,咱們得明白,什么是不可變集合。簡單來說,不可變集合就是一旦創(chuàng)建就不能被修改的集合。
為啥要用不可變集合呢?想象一下,你寫了一段代碼,把一個(gè)集合傳給了別的方法。如果那個(gè)方法不小心改了你的集合,那豈不是一場(chǎng)災(zāi)難?但如果你的集合是不可變的,這種情況就絕對(duì)不會(huì)發(fā)生。不可變集合還有助于編寫更加清晰、更容易維護(hù)的代碼,還能提高程序的性能哦。
第2章:Guava不可變集合簡介
Guava是Google推出的一個(gè)Java庫,里面有一堆好用的工具類,其中就包括了不可變集合。Guava的不可變集合和咱們平時(shí)用的Java標(biāo)準(zhǔn)庫集合有啥不同呢?主要是Guava的集合一旦創(chuàng)建,就不能被修改,這就大大減少了出錯(cuò)的可能性。
來,讓我給你演示一下怎么用Guava創(chuàng)建不可變集合。比如說,咱們要?jiǎng)?chuàng)建一個(gè)不可變的列表:
import com.google.common.collect.ImmutableList; public class ImmutableDemo { public static void main(String[] args) { ImmutableList<String> immutableList = ImmutableList.of("Apple", "Banana", "Cherry"); System.out.println(immutableList); } }
這段代碼創(chuàng)建了一個(gè)包含三種水果名稱的不可變列表。你看,使用ImmutableList.of
就能輕松創(chuàng)建。這樣一來,無論誰拿到這個(gè)列表,都不能添加、刪除或者修改里面的元素。
同理,不可變的集合(Set)和映射(Map)也可以用類似的方式創(chuàng)建。只要記住,一旦創(chuàng)建,就不能更改了。這樣的特性在很多場(chǎng)景下都非常實(shí)用,比如在多線程環(huán)境下,不可變集合可以保證數(shù)據(jù)的安全性,防止出現(xiàn)并發(fā)修改異常。
第3章:創(chuàng)建不可變集合
讓我們來看看不可變列表(ImmutableList)的創(chuàng)建。咱們已經(jīng)看過最基本的創(chuàng)建方式了,現(xiàn)在再來看點(diǎn)高級(jí)的:
import com.google.common.collect.ImmutableList; public class ImmutableListDemo { public static void main(String[] args) { // 創(chuàng)建一個(gè)不可變列表 ImmutableList<String> immutableList = ImmutableList.<String>builder() .add("Apple") .add("Banana") .add("Cherry") .build(); System.out.println(immutableList); } }
在這個(gè)例子中,小黑用了ImmutableList.builder()
方法。這個(gè)方法特別棒,因?yàn)樗梢宰屧蹅円徊讲降靥砑釉兀詈笤儆?code>build()方法一次性創(chuàng)建不可變列表。
接下來,看看不可變集(ImmutableSet)的創(chuàng)建:
import com.google.common.collect.ImmutableSet; public class ImmutableSetDemo { public static void main(String[] args) { // 創(chuàng)建一個(gè)不可變集 ImmutableSet<String> immutableSet = ImmutableSet.of("Apple", "Banana", "Cherry"); System.out.println(immutableSet); } }
ImmutableSet.of()
方法和列表的創(chuàng)建很像,也是一次性把所有元素傳進(jìn)去。
我們來看看不可變映射(ImmutableMap)的創(chuàng)建:
import com.google.common.collect.ImmutableMap; public class ImmutableMapDemo { public static void main(String[] args) { // 創(chuàng)建一個(gè)不可變映射 ImmutableMap<String, Integer> immutableMap = ImmutableMap.of("Apple", 1, "Banana", 2, "Cherry", 3); System.out.println(immutableMap); } }
在這個(gè)例子中,ImmutableMap.of()
方法是用鍵值對(duì)的方式來創(chuàng)建映射的。這里,“Apple”是鍵,1是它的值,以此類推。
第4章:不可變集合的優(yōu)勢(shì)
安全性
首先是安全性。不可變對(duì)象是線程安全的,這意味著在多線程環(huán)境中,你完全不需要擔(dān)心并發(fā)修改的問題。因?yàn)閿?shù)據(jù)一旦創(chuàng)建就不會(huì)改變,所以不會(huì)出現(xiàn)線程間的沖突。這在編寫并發(fā)程序時(shí)特別有用。
舉個(gè)例子吧,假設(shè)有這樣一個(gè)場(chǎng)景:
public class ThreadSafetyDemo { public static void main(String[] args) { ImmutableList<String> immutableList = ImmutableList.of("Apple", "Banana", "Cherry"); // 多線程環(huán)境下訪問不可變列表 Runnable task = () -> { for (String fruit : immutableList) { System.out.println(Thread.currentThread().getName() + " - " + fruit); } }; new Thread(task).start(); new Thread(task).start(); } }
在這個(gè)例子中,即使多個(gè)線程同時(shí)訪問immutableList
,也不會(huì)引發(fā)任何問題,因?yàn)樗遣豢勺兊摹?/p>
效率
接下來談?wù)勑?。不可變?duì)象由于狀態(tài)不變,可以減少內(nèi)存的占用,因?yàn)樗鼈兛梢员蛔杂傻毓蚕怼_@就意味著在一定條件下,不可變集合比可變集合更加內(nèi)存高效。
可讀性
不可變集合還能提高代碼的可讀性和維護(hù)性。當(dāng)你看到一個(gè)集合是不可變的,你就可以立即確定這個(gè)集合在整個(gè)生命周期內(nèi)都不會(huì)改變。這就減少了理解和維護(hù)代碼的復(fù)雜性。
使用Guava的不可變集合可以帶來很多好處,特別是在處理安全性、效率和可讀性方面。雖然它們?cè)谀承┣闆r下可能有些限制,但在正確的場(chǎng)景下使用,絕對(duì)可以幫助你寫出更優(yōu)雅、更穩(wěn)定的Java代碼。
第5章:使用場(chǎng)景與最佳實(shí)踐
使用場(chǎng)景
常量集合:當(dāng)你想定義一些永遠(yuǎn)不變的數(shù)據(jù),比如配置項(xiàng)、選項(xiàng)列表等,不可變集合是最佳選擇。它們可以公開訪問而不用擔(dān)心被意外修改。
public class Constants { public static final ImmutableList<String> FRUITS = ImmutableList.of("Apple", "Banana", "Cherry"); // 其他常量... }
類內(nèi)部狀態(tài):在創(chuàng)建類時(shí),使用不可變集合來保存內(nèi)部狀態(tài)可以確保類的實(shí)例一旦創(chuàng)建就保持不變,這對(duì)于創(chuàng)建不可變類(immutable class)非常有用。
public class UserPreferences { private final ImmutableSet<String> preferences; public UserPreferences(Set<String> preferences) { this.preferences = ImmutableSet.copyOf(preferences); } // Getter方法... }
函數(shù)返回值:當(dāng)函數(shù)需要返回集合時(shí),使用不可變集合作為返回類型可以確保調(diào)用者不會(huì)修改這個(gè)集合,從而保證了數(shù)據(jù)的完整性。
public class ProductService { public ImmutableList<Product> getAvailableProducts() { // 查詢并返回產(chǎn)品列表 } }
最佳實(shí)踐
- 避免預(yù)期外的修改:使用不可變集合可以防止調(diào)用者意外修改集合內(nèi)容,從而導(dǎo)致難以追蹤的bug。
提前拷貝:當(dāng)從可變集合創(chuàng)建不可變集合時(shí),應(yīng)當(dāng)在創(chuàng)建時(shí)就進(jìn)行拷貝,以確保不可變集合的獨(dú)立性。
public class ConfigLoader { public ImmutableSet<String> loadConfig(Set<String> mutableSet) { return ImmutableSet.copyOf(mutableSet); } }
在構(gòu)造函數(shù)中使用不可變集合:當(dāng)創(chuàng)建對(duì)象時(shí),可以使用Guava的不可變集合作為構(gòu)造函數(shù)參數(shù),這樣可以在對(duì)象創(chuàng)建時(shí)就確保其不可變性。
public class Employee { private final ImmutableList<String> skills; public Employee(List<String> skills) { this.skills = ImmutableList.copyOf(skills); } // Getter方法... }
第6章:與Java 8及以后版本的整合
利用Streams處理不可變集合
Java 8的Streams API可以和Guava的不可變集合無縫合作。比如說,你可以輕松地將一個(gè)不可變集合轉(zhuǎn)換成Stream,進(jìn)行各種操作,然后再收集回不可變集合。
import com.google.common.collect.ImmutableList; import java.util.stream.Collectors; public class StreamIntegrationDemo { public static void main(String[] args) { ImmutableList<String> immutableList = ImmutableList.of("Apple", "Banana", "Cherry"); // 使用Stream API處理Guava不可變集合 ImmutableList<String> processedList = immutableList.stream() .filter(s -> s.startsWith("B")) .collect(ImmutableList.toImmutableList()); System.out.println(processedList); } }
在這個(gè)例子里,小黑先把一個(gè)不可變列表轉(zhuǎn)換成了Stream,然后用filter方法篩選出以"B"開頭的元素,最后再收集回一個(gè)不可變列表。這就是Java 8 Stream和Guava不可變集合強(qiáng)強(qiáng)聯(lián)合的例子。
使用Lambda表達(dá)式
Java 8的Lambda表達(dá)式也可以和Guava的不可變集合很好地結(jié)合。它們可以使得對(duì)集合的操作更加簡潔。
import com.google.common.collect.ImmutableList; public class LambdaIntegrationDemo { public static void main(String[] args) { ImmutableList<String> immutableList = ImmutableList.of("Apple", "Banana", "Cherry"); // 使用Lambda表達(dá)式遍歷不可變集合 immutableList.forEach(fruit -> System.out.println("Fruit: " + fruit)); } }
在這個(gè)例子中,咱們使用了forEach方法和Lambda表達(dá)式來遍歷不可變集合。這種方式比傳統(tǒng)的for循環(huán)更簡潔,更易讀。
Guava的不可變集合與Java 8及以后版本的特性相結(jié)合,可以提供更強(qiáng)大的數(shù)據(jù)處理能力,同時(shí)讓代碼變得更加簡潔和易于理解。
第7章:避免常見陷阱
不可變集合不等于只讀集合
首先要清楚,Guava的不可變集合和只讀集合不是一回事。不可變集合是在創(chuàng)建時(shí)就確定了內(nèi)容,而只讀集合只是不能修改,其底層數(shù)據(jù)可能被其他引用修改??磦€(gè)例子:
import com.google.common.collect.ImmutableList; import java.util.ArrayList; import java.util.Collections; import java.util.List; public class ImmutableVsReadOnly { public static void main(String[] args) { List<String> list = new ArrayList<>(); list.add("Apple"); list.add("Banana"); // 創(chuàng)建只讀集合 List<String> readOnlyList = Collections.unmodifiableList(list); // 創(chuàng)建不可變集合 ImmutableList<String> immutableList = ImmutableList.copyOf(list); // 修改原始列表 list.add("Cherry"); // 只讀集合的內(nèi)容發(fā)生了變化 System.out.println("Read-only list: " + readOnlyList); // 不可變集合的內(nèi)容沒變 System.out.println("Immutable list: " + immutableList); } }
這個(gè)例子展示了只讀集合和不可變集合的區(qū)別。當(dāng)原始集合修改時(shí),只讀集合的內(nèi)容也會(huì)跟著變,但不可變集合的內(nèi)容不會(huì)變。
注意構(gòu)建時(shí)的副作用
在使用ImmutableList.builder()
或類似的構(gòu)建器時(shí),要注意不要引入副作用。比如,不要在添加元素的過程中修改這些元素。
// 錯(cuò)誤示范:在構(gòu)建不可變集合時(shí)修改元素 ImmutableList.Builder<String> builder = ImmutableList.builder(); for (String fruit : fruits) { modifyFruit(fruit); // 不應(yīng)在這里修改fruit builder.add(fruit); }
小心空指針異常
Guava的不可變集合在創(chuàng)建時(shí)會(huì)對(duì)元素進(jìn)行非空校驗(yàn),這意味著如果你試圖添加一個(gè)null元素,它會(huì)立即拋出NullPointerException
。
// 這將拋出NullPointerException ImmutableList<String> list = ImmutableList.of("Apple", null, "Cherry");
總結(jié)
選擇不可變集合不僅僅是為了編碼的方便,更重要的是它們提供了額外的安全性、效率和可維護(hù)性。在多線程環(huán)境下,不可變集合幾乎是必不可少的,因?yàn)樗鼈兲焐褪蔷€程安全的。此外,它們還能幫助減少bug和意外行為,尤其是在大型和復(fù)雜的項(xiàng)目中。
要注意,在某些情況下,使用不可變集合可能會(huì)有性能開銷,特別是在需要頻繁修改集合的場(chǎng)景中。因此,選擇合適的數(shù)據(jù)結(jié)構(gòu)和集合類型對(duì)于任何項(xiàng)目都至關(guān)重要。
希望這篇博客能夠幫助你更好地理解Guava的不可變集合,讓你在Java編程的道路上更進(jìn)一步。如果你有任何問題或者想深入討論,歡迎在評(píng)論區(qū)留言,小黑很樂意和你一起探討和學(xué)習(xí)!
以上就是Guava輕松創(chuàng)建和管理不可變集合方法技巧的詳細(xì)內(nèi)容,更多關(guān)于Guava創(chuàng)建管理不可變集合的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
springboot 2.x整合mybatis實(shí)現(xiàn)增刪查和批量處理方式
這篇文章主要介紹了springboot 2.x整合mybatis實(shí)現(xiàn)增刪查和批量處理方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09why在重寫equals時(shí)還必須重寫hashcode方法分享
首先我們先來看下String類的源碼:可以發(fā)現(xiàn)String是重寫了Object類的equals方法的,并且也重寫了hashcode方法2013-10-10解決@MapperScan和@Mapper共存之坑XxxMapper?that?could?not?be?fo
這篇文章主要介紹了解決@MapperScan和@Mapper共存之坑XxxMapper?that?could?not?be?found問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-06-06SpringBoot快速設(shè)置攔截器并實(shí)現(xiàn)權(quán)限驗(yàn)證的方法
本篇文章主要介紹了SpringBoot快速設(shè)置攔截器并實(shí)現(xiàn)權(quán)限驗(yàn)證的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-01-01javaweb學(xué)習(xí)總結(jié)——使用JDBC處理MySQL大數(shù)據(jù)
本篇文章主要介紹了JDBC處理MySQL大數(shù)據(jù),有時(shí)是需要用程序把大文本或二進(jìn)制數(shù)據(jù)直接保存到數(shù)據(jù)庫中進(jìn)行儲(chǔ)存的,非常具有實(shí)用價(jià)值,需要的朋友可以參考下。2016-11-11