Java8 Collectors.toMap()和Collectors.groupingBy()输出乱序 |
您所在的位置:网站首页 › hashmap怎么有序 › Java8 Collectors.toMap()和Collectors.groupingBy()输出乱序 |
1. Collectors.toMap() 输出乱序
1.1 场景
想按创建时间降序列表展示订单信息,但最终返回给前端的数据和idList顺序不一致,乱序输出。 Debug发现有段代码,根据idList从数据库中查询出orderList,输出一个以订单编号为key,订单内容为value的Map,该Map输出内容是乱序的。 ...... //根据订单idList查询订单列表 List orderList = orderDao.listOrderByIds(idList); //输出以订单id为key的Map Map orderMap = orders.stream().collect(Collectors.toMap(Order::getId, order -> order, (k1, k2) -> k1)); ......查看Collectors.toMap()源码发现其输出的Map是HashMap,而HashMap不是按顺序存的。 1.2 Collectors.toMap()Collectors.toMap()有三个参数,以Collectors.toMap(Order::getId, a -> a, (k1, k2) -> k1)为例,第一个参数Order::getId为Map的key,第二个参数order -> order为value,第三个参数(k1, k2) -> k1表示出现相同的key时,取旧key值对应的value值。 点进Collectors.toMap()源码,可以看到,输出的Map是HashMap。 public static Collector toMap(Function keyMapper, Function valueMapper, BinaryOperator mergeFunction) { return toMap(keyMapper, valueMapper, mergeFunction, HashMap::new); }而HashMap输出是乱序的。 1.3 HashMapHashMap输出乱序,和它存数据的方式有关。 HashMap使用哈希表来存储,为了解决Hash冲突使用的是链地址法。在每个数组元素上都一个链表结构,当数据被Hash后,得到被扰乱的hash值,然后通过位与运算获取数组下标,最后把数据放在该数组下标链表上。 遍历输出时,是遍历的tab[]数组,而数组经过计算hash值,存进的数据和最初的List顺序可能不一致了。 final V putVal(int hash, K key, V value, boolean onlyIfAbsent, boolean evict) { Node[] tab; Node p; int n, i; if ((tab = table) == null || (n = tab.length) == 0) n = (tab = resize()).length; //1 if ((p = tab[i = (n - 1) & hash]) == null) tab[i] = newNode(hash, key, value, null); else { ...... } ++modCount; if (++size > threshold) resize(); afterNodeInsertion(evict); return null; }计算hash值。 static final int hash(Object key) { int h; //2 return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16); }HashMap遍历Node[] tab输出。 @Override public void forEach(BiConsumer action) { Node[] tab; if (action == null) throw new NullPointerException(); if (size > 0 && (tab = table) != null) { int mc = modCount; for (int i = 0; i if (action == null) throw new NullPointerException(); int mc = modCount; //4 for (LinkedHashMap.Entry e = head; e != null; e = e.after) action.accept(e.key, e.value); if (modCount != mc) throw new ConcurrentModificationException(); }为什么有序参考这篇文章:LinkedHashMap 如何保证插入顺序的(jdk8) 1.5 解决方案为保证输出有序,选择LinkedHashMap,具体修改方案如下: LinkedHashMap orderMap = orders.stream().collect(Collectors.toMap(Order::getId, order -> order, (k1, k2) -> k1,LinkedHashMap::new)); 2. Collectors.groupingBy() 输出乱序 2.1 场景想要以id分组输出Map,结果输出是乱序的。 Map map; map = list.stream().collect(Collectors.groupingBy(OrderVO::getId));其原因和Collectors.toMap()类似,查看源码可知Collectors.groupingBy()输出为HashMap。 public static Collector groupingBy(Function classifier, Collector downstream) { return groupingBy(classifier, HashMap::new, downstream); } 2.2 解决方案选择以LinkedHashMap输出。 map = list.stream().collect(Collectors.groupingBy(OrderVO::getId, LinkedHashMap::new, Collectors.toList())); |
今日新闻 |
推荐新闻 |
CopyRight 2018-2019 办公设备维修网 版权所有 豫ICP备15022753号-3 |