Java Collection 移除元素方法及注意事項(xiàng)
1. 前言
操作集合是一個(gè) Java 編程人員幾乎每天都在重復(fù)的事情。今天我們來(lái)研究一下從 Java Collection 中刪除元素的方法。我構(gòu)建了一個(gè)簡(jiǎn)單的集合,我們以此為例子來(lái)展開(kāi)探索。
List<String> servers = new ArrayList<>();
servers.add("Felordcn");
servers.add("Tomcat");
servers.add("Jetty");
servers.add("Undertow");
servers.add("Resin");
2. for 循環(huán)并不一定能從集合中移除元素
讓我們使用傳統(tǒng)的 foreach 循環(huán)移除 F 開(kāi)頭的假服務(wù)器,但是你會(huì)發(fā)現(xiàn)這種操作引發(fā)了 ConcurrentModificationException 異常。
// 錯(cuò)誤的示范 千萬(wàn)不要使用
for (String server : servers) {
if (server.startsWith("F")) {
servers.remove(server);
}
}
難道 for 循環(huán)就不能移除元素了嗎?當(dāng)然不是!我們?nèi)绻艽_定需要被移除的元素的索引還是可以的。
// 這種方式是可行
for (int i = 0; i < servers.size(); i++) {
if (servers.get(i).startsWith("F")) {
servers.remove(i);
}
}
但是這種方式我目前只演示了 ArrayList,其它的類型并沒(méi)有嚴(yán)格測(cè)試,留給你自己探索。
3. 迭代器 Iterator 可以刪除集合中的元素
在傳統(tǒng)方式中我們使用 Iterator 是可以保證刪除元素的:
Iterator<String> iterator = servers.iterator();
while (iterator.hasNext()) {
String next = iterator.next();
if (next.startsWith("F")) {
iterator.remove();
}
}
4. 遍歷刪除元素的缺點(diǎn)
- 我們需要遍歷集合的每一個(gè)元素并對(duì)它們進(jìn)行斷言,哪怕你刪除一個(gè)元素。
- 盡管我們可以通過(guò)迭代的方式刪除特定的元素,但是操作繁瑣,根據(jù)集合類型的不同有潛在的 ConcurrentModificationException 異常。
- 根據(jù)數(shù)據(jù)結(jié)構(gòu)的不同,刪除元素的時(shí)間復(fù)雜度也大大不同。比如數(shù)組結(jié)構(gòu)的 ArrayList 在刪除元素的速度上不如鏈表結(jié)構(gòu)的 LinkedList。
5. 新的集合元素刪除操作
Java 8 提供了新的集合操作 API 和 Stream 來(lái)幫助我們解決這個(gè)問(wèn)題。我在以前的文章中已經(jīng)介紹了 Java 8 Stream API,如果有興趣可以去看看。
5.1 Collection.removeIf()
新的 Collection Api removeIf(Predicate<? super E> filter) 。該 Api 提供了一種更簡(jiǎn)潔的使用 Predicate (斷言)刪除元素的方法,于是我們可以更加簡(jiǎn)潔的實(shí)現(xiàn)開(kāi)始的需求:
servers.removeIf(s-> s.startsWith("F"));
同時(shí)根據(jù)測(cè)試,ArrayList 和 LinkedList 的性能接近。一般推薦使用這種方式進(jìn)行操作。
5.2 Stream 實(shí)現(xiàn)移除元素
和上面所有移除操作不同的是,其實(shí)任何操作都不會(huì)改變 Stream 源,我們僅僅是使用 Stream Api 操作數(shù)據(jù)源的副本。遵循了 數(shù)據(jù)源 -> 中間操作 -> 歸納終止 的生命周期。我們來(lái)看看使用 Stream 如何實(shí)現(xiàn)我們的意圖。
5.2.1 通過(guò) filter 斷言實(shí)現(xiàn)
我們可以使用 Stream 的 filter 斷言。filter 斷言會(huì)把符合斷言的流元素匯集成一個(gè)新的流,然后歸納起來(lái)即可,于是我們可以這么寫(xiě):
// 跟以上不同的是 該方式中的斷言是取反的操作。
List<String> newServers = servers.stream().filter(s -> !s.startsWith("F")).collect(Collectors.toList());
這個(gè)優(yōu)點(diǎn)上面已經(jīng)說(shuō)了不會(huì)影響原始數(shù)據(jù),生成的是一個(gè)副本。缺點(diǎn)就是可能會(huì)有內(nèi)存占用問(wèn)題。
5.2.2 通過(guò) Collectors.partitioningBy 歸納
這種方法雖然可以滿足需要但是我感覺(jué)有點(diǎn)投機(jī)取巧的成份。Collectors.partitioningBy() 方法本意是做二分類的。該方法會(huì)將流中符合斷言的、不符合斷言的元素分別歸納到兩個(gè) key 分別為 true 和 false 的 Map 中,我們可以歸類得到符合和不符合的元素集。實(shí)現(xiàn)如下:
Map<Boolean, List<String>> f = servers.stream().collect(Collectors.partitioningBy(s -> !s.startsWith("F")));
List<String> trues = f.get(Boolean.TRUE);
System.out.println("不以 F 開(kāi)頭的: " + trues);
List<String> falses = f.get(Boolean.FALSE);
System.out.println("以 F 開(kāi)頭的: " + falses);
一般該方式不推薦在此場(chǎng)景使用,并不符合該 Api 的設(shè)計(jì)意圖。
6. 總結(jié)
今天我們研究了一些從 Collections 中刪除元素的方法 及其注意事項(xiàng)。希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
如果你覺(jué)得本文對(duì)你有幫助,歡迎轉(zhuǎn)載,煩請(qǐng)注明出處,謝謝!
相關(guān)文章
java中的類型自動(dòng)轉(zhuǎn)換機(jī)制解析
這篇文章主要介紹了java中的類型自動(dòng)轉(zhuǎn)換機(jī)制,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03
一文掌握Spring Cookie和Session 是什么及區(qū)別介紹
Cookie和Session都是用于在客戶端和服務(wù)器之間傳遞信息的技術(shù),但它們的工作方式和使用場(chǎng)景有所不同,Cookie是在客戶端保存用戶信息的一種機(jī)制,而Session是在服務(wù)器端保存用戶信息的一種機(jī)制,本文介紹Spring Cookie和Session 是什么,感興趣的朋友一起看看吧2025-01-01
eclipse/intellij idea 查看java源碼和注釋方法
下面小編就為大家?guī)?lái)一篇eclipse/intellij idea 查看java源碼和注釋方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-05-05
Spring?Boot?項(xiàng)目中?JPA?語(yǔ)法的基本使用方法
這篇文章主要介紹了?Spring?Boot?項(xiàng)目中?JPA?語(yǔ)法的基本使用方法,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-10-10
SpringBoot集成FastDFS依賴實(shí)現(xiàn)文件上傳的示例
這篇文章主要介紹了SpringBoot集成FastDFS依賴實(shí)現(xiàn)文件上傳,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05
Java動(dòng)態(tài)線程池插件dynamic-tp集成過(guò)程淺析
這篇文章主要介紹了Java動(dòng)態(tài)線程池插件dynamic-tp集成過(guò)程,dynamic-tp是一個(gè)輕量級(jí)的動(dòng)態(tài)線程池插件,它是一個(gè)基于配置中心的動(dòng)態(tài)線程池,線程池的參數(shù)可以通過(guò)配置中心配置進(jìn)行動(dòng)態(tài)的修改2023-03-03
SVN出現(xiàn)提示org.apache.subversion.javahl.ClientException: Attempt
這篇文章主要介紹了SVN出現(xiàn)提示org.apache.subversion.javahl.ClientException: Attempted to lock an already-locked dir解決方案的相關(guān)資料,需要的朋友可以參考下2016-12-12
SpringBoot實(shí)現(xiàn)子類的反序列化示例代碼
這篇文章主要給大家介紹了關(guān)于SpringBoot實(shí)現(xiàn)子類的反序列化的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用SpringBoot具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08

