java Map(HashMap)的5种遍历方式解析

package com.websocket;

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

/**
 * Created by xyc on 2017/4/9 0009.
 */
public class MapDemo {
    public static void main(String[] args) {
        Map<Integer, Object> map = new HashMap<>();

        int n = 10000000;
        while (n > 0) {
            map.put(n, n);
            n--;
        }

        test1(map); //entrySet:133 137 121
        test2(map); //values:118 114 116
        test3(map); //iterator:122 129 117
        test4(map); //keySet:183 197 185
        test5(map); //forEach:165 227 245
    }

    /**
     * 在java5中被引入所以该方法只能应用于java 5或更高的版本中。
     * 如果你遍历的是一个空的map对象,将抛出NullPointerException,因此在遍历前你总是应该检查空引用。
     * entrySet只是遍历了一次就把key和value都放到了entry中,效率相比keySet更高。
     * entrySet()返回的是K-V值组合集合
     * 执行时间:133 137 121
     *
     * @param map
     */
    public static void test1(Map<Integer, Object> map) {
        long before = System.currentTimeMillis();
        for (Map.Entry<Integer, Object> entry : map.entrySet()) {
            entry.getKey();
            entry.getValue();
        }
        long after = System.currentTimeMillis();
        System.out.println(after - before);
    }

    /**
     * 如果只需要map中的键或者值,你可以通过keySet或values来实现遍历,而不是用entrySet。
     * 该方法比entrySet遍历在性能上稍好(快了10%),而且代码更加干净。
     * values()返回的是V值集合,是一个list集合对象
     * 执行时间:118 114 116
     *
     * @param map
     */
    public static void test2(Map<Integer, Object> map) {
        long before = System.currentTimeMillis();
        for (Object value : map.values()) {
        }
        long after = System.currentTimeMillis();
        System.out.println(after - before);
    }

    /**
     * 该种方式看起来冗余却有其优点所在。首先,在老版本java中这是惟一遍历map的方式。
     * 另一个好处是,你可以在遍历时调用iterator.remove()来删除entries,另两个方法则不能。从性能方面看,该方法类同于keySet和values的性能。
     * 执行时间:122 129 117
     *
     * @param map
     */
    public static void test3(Map<Integer, Object> map) {
        long before = System.currentTimeMillis();
        Iterator<Map.Entry<Integer, Object>> entries = map.entrySet().iterator();
        while (entries.hasNext()) {
            Map.Entry<Integer, Object> entry = entries.next();
            entry.getKey();
            entry.getValue();
        }
        long after = System.currentTimeMillis();
        System.out.println(after - before);
    }

    /**
     * 作为entrySet的替代,这个代码看上去更加干净;但实际上它相当慢且无效率。
     * 因为从键取值是耗时的操作(与entrySet相比,在不同的Map实现中该方法慢了20%~200%)。
     * keySet其实是遍历了2次,一次是转为Iterator对象,另一次是从hashMap中取出key所对应的value。
     * keySet()返回的是K值集合,是一个Set集合对象
     * 执行时间:183 197 185
     *
     * @param map
     */
    public static void test4(Map<Integer, Object> map) {
        long before = System.currentTimeMillis();
        for (Integer key : map.keySet()) {
            Object value = map.get(key);
        }
        long after = System.currentTimeMillis();
        System.out.println(after - before);
    }

    /**
     * forEach的内部其实是对entrySet的封装,性能在五种方法中是最低的。
     * 执行时间:165 227 245
     *
     * @param map
     */
    public static void test5(Map<Integer, Object> map) {
        long before = System.currentTimeMillis();
        map.forEach((key, value) -> {
        });
        long after = System.currentTimeMillis();
        System.out.println(after - before);
    }
}
总结
  • 如果仅需要键(keys)则使用test4;
  • 如果仅需要值(values)使用test2;
  • 如果你使用的语言版本低于java 5,或是打算在遍历时删除entries,必须使用test3;
  • 如果你使用的语言版本JDK8,则可以使用test5;
  • 否则可以使用test1(键值都要);

参考:http://blog.csdn.net/tjcyjd/article/details/11111401

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值