默认groupingBy代码里会生成一个HashMap(hashMap是无序的,put的顺序与get的顺序不一致)spa
- 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的顺序是不可控的.code
如今遇到这样一个场景排序
在CMS里,每一个页面的模块是按顺序排放的,每一个模块的内容也是按顺序的如 get
List<Model> list=Arrays.asList(m1,m2,m3)hash
如今须要对里面的元素分组,可是分组后的顺序也必须是 m1,m2,m3...中间能够缺乏,可是不能乱序io
如下是合法的 m1,m3 或者 m2,m3 可是不能 m3,m2class
如如下代码 list的顺序是 id=2的在 id=1以前, 分组以后的访问也必须是id=2的在前才对容器
可是若是调用 默认的分组,就会发现 id=1的在前了 (在后的将要在前;在前的将要在后了)stream
输出老是List
1
[A12,A11]
2
[A2,A21]
可是指望输出为
2 [A21,A2] 1 [A12,A11]
若是须要保持排序就不能使用默认的 方法了,必须使用 被注释的方法 (明确使用LinkedHashMap来保持顺序).
下面是groupingBy的参数说明
能够看到有三个参数,第一个参数就是key的Function
了,第二个参数是一个map工厂,也就是最终结果的容器,通常默认的是采用的HashMap::new
,最后一个参数很重要是一个downstream
,类型是Collector
,也是一个收集器,那就是说,这三个参数其实就是为了解决分组问题的
-
第一个参数:分组按照什么分类
-
第二个参数:分组最后用什么容器保存返回
-
第三个参数:按照第一个参数分类后,对应的分类的结果如何收集
其实一个参数的Collectors.groupingBy
方法的 ,第二个参数默认是HashMap::new
, 第三个参数收集器其实默认是Collectors.toList