您的位置:首页 > 编程语言 > Java开发

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

2017-04-09 18:11 381 查看
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
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  java hashmap 遍历 map