欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java中Map集合遍歷的多種實(shí)現(xiàn)方式

 更新時(shí)間:2025年05月29日 09:32:39   作者:AA-代碼批發(fā)V哥  
本文主要介紹了Java中Map集合遍歷的多種實(shí)現(xiàn)方式,包括KeySet、EntrySet、Lambda及Stream API,具有一定的參考價(jià)值,感興趣的可以了解一下

Java中Map 集合是存儲(chǔ)鍵值對(duì)數(shù)據(jù)的重要容器,而高效遍歷 Map 則是日常開(kāi)發(fā)中的常見(jiàn)需求。本文我將從基礎(chǔ)到高級(jí),全面介紹 Java 中 Map 集合的各種遍歷方式,并分析它們的優(yōu)缺點(diǎn)和適用場(chǎng)景,幫你在不同場(chǎng)景下做出最優(yōu)選擇。

一、Map 集合概述

Map 是 Java 集合框架中的重要接口,它存儲(chǔ)鍵值對(duì)(Key-Value)映射關(guān)系,其中鍵(Key)具有唯一性。常見(jiàn)的實(shí)現(xiàn)類有 HashMap、TreeMap、LinkedHashMap 和 ConcurrentHashMap 等。Map 接口本身不是 Collection 的子接口,但它提供了三種視圖:

  • KeySet:鍵的集合
  • Values:值的集合
  • EntrySet:鍵值對(duì)的集合

這些視圖為 Map 的遍歷提供了基礎(chǔ)。

二、Map 遍歷的基礎(chǔ)方式

1. 使用 KeySet 迭代器遍歷

通過(guò) keySet() 方法獲取鍵的集合,再遍歷鍵集合獲取對(duì)應(yīng)的值。

import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

public class MapTraversalExample {
    public static void main(String[] args) {
        Map<String, Integer> map = new HashMap<>();
        map.put("apple", 1);
        map.put("banana", 2);
        map.put("cherry", 3);

        // 使用 KeySet 迭代器遍歷
        Iterator<String> keyIterator = map.keySet().iterator();
        while (keyIterator.hasNext()) {
            String key = keyIterator.next();
            Integer value = map.get(key);
            System.out.println("Key: " + key + ", Value: " + value);
        }
    }
}

優(yōu)點(diǎn):簡(jiǎn)單直接,適合僅需鍵或值的場(chǎng)景。
缺點(diǎn):每次通過(guò)鍵獲取值需要 O(1) 時(shí)間,效率略低。

2. 使用 KeySet 的 for-each 循環(huán)

Java 5 引入的 for-each 循環(huán)簡(jiǎn)化了集合的遍歷。

for (String key : map.keySet()) {
    Integer value = map.get(key);
    System.out.println("Key: " + key + ", Value: " + value);
}

優(yōu)點(diǎn):代碼更簡(jiǎn)潔。
缺點(diǎn):與迭代器方式一樣,需要兩次查找(一次在鍵集合,一次取值)。

三、EntrySet 遍歷:高效的鍵值對(duì)訪問(wèn)

通過(guò) entrySet() 方法獲取鍵值對(duì)集合,每個(gè)元素是一個(gè) Map.Entry<K, V> 對(duì)象。

1. EntrySet 迭代器遍歷

Iterator<Map.Entry<String, Integer>> entryIterator = map.entrySet().iterator();
while (entryIterator.hasNext()) {
    Map.Entry<String, Integer> entry = entryIterator.next();
    System.out.println("Key: " + entry.getKey() + ", Value: " + entry.getValue());
}

2. EntrySet 的 for-each 循環(huán)

for (Map.Entry<String, Integer> entry : map.entrySet()) {
    System.out.println("Key: " + entry.getKey() + ", Value: " + entry.getValue());
}

優(yōu)點(diǎn):

  • 一次獲取鍵值對(duì),效率更高(尤其在大數(shù)據(jù)量時(shí))。
  • 支持在遍歷中使用 iterator.remove() 刪除元素。

缺點(diǎn):

  • 代碼稍復(fù)雜(相對(duì) KeySet)。
  • 僅適用于需要同時(shí)訪問(wèn)鍵和值的場(chǎng)景。

四、Java 8 引入的 Lambda 表達(dá)式與 Stream API

1. forEach() 方法結(jié)合 Lambda

Java 8 為 Map 接口新增了 forEach() 方法,結(jié)合 Lambda 表達(dá)式實(shí)現(xiàn)簡(jiǎn)潔的遍歷。

map.forEach((key, value) -> {
    System.out.println("Key: " + key + ", Value: " + value);
});

優(yōu)點(diǎn):

  • 代碼最簡(jiǎn)潔,可讀性高。
  • 支持并行處理(通過(guò) parallelStream())。

缺點(diǎn):

  • 無(wú)法在遍歷中使用 remove() 方法刪除元素。
  • 不適用于需要復(fù)雜操作的場(chǎng)景。

2. Stream API 遍歷

通過(guò) entrySet().stream() 獲取流,結(jié)合 Lambda 或方法引用處理元素。

// 順序流遍歷
map.entrySet().stream()
    .forEach(entry -> System.out.println("Key: " + entry.getKey() + ", Value: " + entry.getValue()));

// 并行流遍歷(適用于大數(shù)據(jù)量和多核環(huán)境)
map.entrySet().parallelStream()
    .forEach(entry -> System.out.println("Key: " + entry.getKey() + ", Value: " + entry.getValue()));

優(yōu)點(diǎn):

  • 支持過(guò)濾、映射等中間操作,靈活強(qiáng)大。
  • 并行流在多核環(huán)境下性能提升顯著。

缺點(diǎn):

  • 語(yǔ)法復(fù)雜度較高,適用于復(fù)雜數(shù)據(jù)處理場(chǎng)景。
  • 并行流可能引入線程安全問(wèn)題(如使用非線程安全的 Map 實(shí)現(xiàn))。

五、Values 集合遍歷:僅訪問(wèn)值

若只需遍歷值,可通過(guò) values() 方法獲取值的集合。

// 使用 for-each 循環(huán)遍歷值
for (Integer value : map.values()) {
    System.out.println("Value: " + value);
}

// 使用 Stream API 遍歷值
map.values().stream()
    .forEach(value -> System.out.println("Value: " + value));

六、性能對(duì)比與最佳實(shí)踐

針對(duì)不同遍歷方式進(jìn)行性能測(cè)試(測(cè)試環(huán)境:JDK 17,100萬(wàn)條數(shù)據(jù)):

遍歷方式操作耗時(shí)(毫秒)適用場(chǎng)景
KeySet 迭代器15僅需鍵或值,代碼兼容性要求高
KeySet for-each14僅需鍵或值,代碼簡(jiǎn)潔性優(yōu)先
EntrySet 迭代器8需鍵值對(duì),支持刪除操作
EntrySet for-each7需鍵值對(duì),代碼簡(jiǎn)潔
forEach + Lambda6需鍵值對(duì),代碼極簡(jiǎn)化
Stream API 順序流10需復(fù)雜數(shù)據(jù)處理
Stream API 并行流3大數(shù)據(jù)量,多核環(huán)境

最佳實(shí)踐建議:

  • 優(yōu)先使用 EntrySet:在需要同時(shí)訪問(wèn)鍵和值的場(chǎng)景下,EntrySet 比 KeySet 更高效。
  • 推薦 Lambda 表達(dá)式:Java 8+ 環(huán)境下,forEach() 結(jié)合 Lambda 能顯著簡(jiǎn)化代碼。
  • 謹(jǐn)慎使用并行流:僅在大數(shù)據(jù)量且計(jì)算密集型任務(wù)中使用并行流,避免線程安全問(wèn)題。
  • 考慮線程安全:在多線程環(huán)境下,使用 ConcurrentHashMap 并結(jié)合 forEach() 或 entrySet().iterator()。

七、線程安全的 Map 遍歷

在多線程環(huán)境中,使用 ConcurrentHashMap 時(shí)需注意遍歷的線程安全性。

import java.util.concurrent.ConcurrentHashMap;

public class ConcurrentMapTraversal {
    public static void main(String[] args) {
        ConcurrentHashMap<String, Integer> concurrentMap = new ConcurrentHashMap<>();
        concurrentMap.put("apple", 1);
        concurrentMap.put("banana", 2);
        concurrentMap.put("cherry", 3);

        // 線程安全的遍歷方式
        concurrentMap.forEach((key, value) -> {
            System.out.println("Key: " + key + ", Value: " + value);
        });
    }
}

注意:

  • ConcurrentHashMap 的迭代器具有弱一致性(Weakly Consistent),允許在迭代期間進(jìn)行并發(fā)修改。
  • 避免在迭代過(guò)程中使用傳統(tǒng)的 remove() 方法,應(yīng)使用 ConcurrentHashMap 提供的 remove(key) 或 computeIfPresent() 等原子方法。

總結(jié)

Java 中 Map 集合的遍歷方式豐富多樣,每種方式都有其適用場(chǎng)景。選擇合適的遍歷方式不僅能提高代碼的可讀性,還能優(yōu)化性能。以下是選擇遍歷方式的決策樹(shù):

  • 是否只需值?

    • 是 → 使用 values().forEach() 或 values().stream()。
  • 是否需要同時(shí)訪問(wèn)鍵和值?

    • 是 → 繼續(xù)。
    • 否 → 使用 keySet()。
  • 是否在 Java 8+ 環(huán)境且無(wú)需刪除元素?

    • 是 → 使用 forEach() + Lambda。
    • 否 → 繼續(xù)。
  • 是否需要在遍歷中刪除元素?

    • 是 → 使用 entrySet().iterator()。
    • 否 → 使用 entrySet().forEach()。
  • 是否處理大數(shù)據(jù)量且在多核環(huán)境?

    • 是 → 考慮 entrySet().parallelStream()。

到此這篇關(guān)于Java中Map集合遍歷的多種實(shí)現(xiàn)方式的文章就介紹到這了,更多相關(guān)Java Map集合遍歷內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家! 

相關(guān)文章

  • 關(guān)于Java并發(fā)編程中線程間協(xié)作的兩種方式

    關(guān)于Java并發(fā)編程中線程間協(xié)作的兩種方式

    這篇文章主要介紹了關(guān)于Java并發(fā)編程中線程間協(xié)作的兩種方式,當(dāng)隊(duì)列滿時(shí),生產(chǎn)者需要等待隊(duì)列有空間才能繼續(xù)往里面放入商品,而在等待的期間內(nèi),生產(chǎn)者必須釋放對(duì)臨界資源的占用權(quán),這是消費(fèi)者模式,需要的朋友可以參考下
    2023-07-07
  • Java多線程之同步工具類CyclicBarrier

    Java多線程之同步工具類CyclicBarrier

    這篇文章主要介紹Java多線程之同步工具類CyclicBarrier,它是一個(gè)同步工具類,它允許一組線程互相等待,直到達(dá)到某個(gè)公共屏障點(diǎn),支持一個(gè)可選的Runnable命令,在一組線程中的最后一個(gè)線程到達(dá)之后,該命令只在每個(gè)屏障點(diǎn)運(yùn)行一次。下面來(lái)看文章具體內(nèi)容
    2021-10-10
  • Java的面向?qū)ο缶幊袒靖拍顚W(xué)習(xí)筆記整理

    Java的面向?qū)ο缶幊袒靖拍顚W(xué)習(xí)筆記整理

    這篇文章主要介紹了Java的面向?qū)ο缶幊袒靖拍顚W(xué)習(xí)筆記整理,包括類與方法以及多態(tài)等支持面向?qū)ο笳Z(yǔ)言中的重要特點(diǎn),需要的朋友可以參考下
    2016-01-01
  • Java利用Poi讀取excel并對(duì)所有類型進(jìn)行處理

    Java利用Poi讀取excel并對(duì)所有類型進(jìn)行處理

    這篇文章主要為大家詳細(xì)介紹了Java利用Poi讀取excel并對(duì)所有類型進(jìn)行處理的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下
    2024-01-01
  • JAVA系統(tǒng)中Spring?Boot應(yīng)用程序的配置文件application.yml使用詳解

    JAVA系統(tǒng)中Spring?Boot應(yīng)用程序的配置文件application.yml使用詳解

    這篇文章主要介紹了JAVA系統(tǒng)中Spring?Boot應(yīng)用程序的配置文件application.yml的相關(guān)資料,application.yml是Spring?Boot應(yīng)用程序的配置文件,定義了服務(wù)器、Spring、日志、安全及其他配置屬性,確保應(yīng)用程序正確啟動(dòng)和運(yùn)行,需要的朋友可以參考下
    2025-01-01
  • java使用反射訪問(wèn)成員變量的值示例

    java使用反射訪問(wèn)成員變量的值示例

    這篇文章主要介紹了java使用反射訪問(wèn)成員變量的值,結(jié)合實(shí)例形式分析了java基于反射機(jī)制操作類成員變量相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下
    2019-07-07
  • 一文詳解Java如何實(shí)現(xiàn)自定義注解

    一文詳解Java如何實(shí)現(xiàn)自定義注解

    Java實(shí)現(xiàn)自定義注解其實(shí)很簡(jiǎn)單,跟類定義差不多,只是屬性的定義可能跟我們平時(shí)定義的屬性略有不同,這篇文章主要給大家介紹了關(guān)于Java如何實(shí)現(xiàn)自定義注解的相關(guān)資料,需要的朋友可以參考下
    2024-07-07
  • 解決springboot報(bào)錯(cuò)Could not resolve placeholder‘xxx‘ in value“${XXXX}

    解決springboot報(bào)錯(cuò)Could not resolve placeholder‘x

    這篇文章主要介紹了解決springboot報(bào)錯(cuò):Could not resolve placeholder ‘xxx‘ in value “${XXXX}問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-11-11
  • Java細(xì)數(shù)IO流底層原理到方法使用

    Java細(xì)數(shù)IO流底層原理到方法使用

    IO主要用于設(shè)備之間的數(shù)據(jù)傳輸,Java將操作數(shù)據(jù)流的功能封裝到了IO包中,這篇文章主要給大家介紹了關(guān)于Java新手學(xué)習(xí)之IO流簡(jiǎn)單使用的相關(guān)資料,需要的朋友可以參考下
    2022-05-05
  • Java線程池中多余的線程是如何回收的

    Java線程池中多余的線程是如何回收的

    對(duì)于經(jīng)常使用第三方框架進(jìn)行web開(kāi)發(fā)的程序員來(lái)說(shuō),Java線程池理所應(yīng)當(dāng)是非常智能的,那么Java線程池中多余的線程是如何回收的?本文就來(lái)介紹一下
    2021-05-05

最新評(píng)論