【java8】 java8分组后,对数据顺序不改变的控制

   一.概述

 默认groupingBy代码里会生成一个HashMap(hashMap是无序的,put的顺序与get的顺序不一致)

  • HashMap是无序的,HashMap在put的时候是根据key的hashcode进行hash然后放入对应的地方。所以在按照一定顺序put进HashMap中,然后遍历出HashMap的顺序跟put的顺序不同(除非在put的时候key已经按照hashcode排序号了,这种几率非常小)
  • 单纯的HashMap是无法实现排序的,这的排序是指,我们将键值对按照一定的顺序put进HashMap里,然后在进行取键值对的操作的时候,是按照put进去的顺序把键值对取出来的。
  • JAVA在JDK1.4以后提供了LinkedHashMap来帮助我们实现了有序的HashMap!LinkedHashMap取键值对时,是按照你放入的顺序来取的。

二.结果

这就造成了一个List<Model>如果是有序的,在 groupingBy后 model的顺序是不可控的.

三.解决

下面是groupingBy的参数说明

可以看到有三个参数,第一个参数就是key的Function了,第二个参数是一个map工厂,也就是最终结果的容器,一般默认的是采用的HashMap::new,最后一个参数很重要是一个downstream,类型是Collector,也是一个收集器,那就是说,这三个参数其实就是为了解决分组问题的

  • 第一个参数:分组按照什么分类

  • 第二个参数:分组最后用什么容器保存返回

  • 第三个参数:按照第一个参数分类后,对应的分类的结果如何收集

其实一个参数的Collectors.groupingBy方法的 ,第二个参数默认是HashMap::new, 第三个参数收集器其实默认是Collectors.toList

图例:

连续分组加排序:

  1. LinkedHashMap<String,LinkedHashMap<String,List<ProvinceAndCityAndDistrict>>> resMap = target.stream()
  2. .collect(Collectors.groupingBy(provinceAndCityAndDistrict::getsProvince,LinkedHashMap::new,Collectors.groupingBy(ProvinceAndCityAndDistrict::getsCity,LinkedHashMap::new,Collectors.toList())));

 第一个参数:分组参数,key值

 第二个参数:分组后甩什么容器保存,指定为lindedHashMap

 第三个参数:按照第一个参数分组后,对应的分类结果再次分组,并且最后用list进行收集

 

 

 

Java中,如果你有一个`List<Map<String, Object>>`类型的集合,其中包含`id`和`date`等键值对,并希望根据`id`进行排,然后按照`date`字段进行分组,同时保持每个分组内的顺序变,你可以这样做: 首先,你需要创建一个自定义的Comparator来比较Map,这里假设`id`是数字类型,可以自然排,而`date`通常是一个Date对象,可以先将其转换成String以便于排。 ```java import java.text.SimpleDateFormat; import java.util.*; class MapComparator implements Comparator<Map<String, Object>> { private SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd"); @Override public int compare(Map<String, Object> map1, Map<String, Object> map2) { Long id1 = (Long) map1.get("id"); Long id2 = (Long) map2.get("id"); // 先按id排 int idCompare = Integer.compare(id1, id2); if (idCompare != 0) return idCompare; // 如果id相等,则按照日期排,这里假设date都是字符串形式 String dateStr1 = dateFormat.format(map1.get("date")); String dateStr2 = dateFormat.format(map2.get("date")); return dateStr1.compareTo(dateStr2); } } ``` 接着,你可以使用Collections.sort()方法并配合自定义的Comparator来完成排分组: ```java List<Map<String, Object>> sortedAndGrouped = new ArrayList<>(); // 假设你已经填充了mapList MapComparator comparator = new MapComparator(); Collections.sort(sortedAndGrouped, comparator); // 使用Stream API进行分组,但注意Java Stream的groupingBy会改变元素顺序 Map<Long, List<Map<String, Object>>> groupedMaps = sortedAndGrouped.stream() .collect(Collectors.groupingBy(map -> (long) map.get("id"), Collectors.toList())); ``` 现在,`groupedMaps`中存储的就是按照id排并且在每个id内按date排后的数据。请注意,由于`Collectors.toList()`默认的内部排规则,列表内的顺序可能会与输入的原始顺序略有差异,如果需要保证原顺序,可以考虑使用`LinkedHashMap`作为分组容器。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值