Java的Stream流,groupingBy之后无序解决办法

当使用JavaStream的groupingBy函数对List集合进行分组时,由于HashMap的无序性,分组后的顺序可能与原数据顺序不同。解决这个问题的方法是使用带三个参数的groupingBy函数,传入LinkedHashMap作为mapFactory,以保持分组后的顺序。
摘要由CSDN通过智能技术生成

问题
一般情况下,我们拿到List集合数据后是有序的,通过stream的groupingBy函数分组之后,顺序却被打乱了,并没有按照list里面的顺序进行返回。

解决
通过源码可以知道,Collectors类里面有3个groupingBy函数;

    // 第一个
    public static <T, K> Collector<T, ?, Map<K, List<T>>>
    groupingBy(Function<? super T, ? extends K> classifier) {
        return groupingBy(classifier, toList());
    }

    // 第二个
    public static <T, K, A, D>
    Collector<T, ?, Map<K, D>> groupingBy(Function<? super T, ? extends K> classifier,
                                          Collector<? super T, A, D> downstream) {
        return groupingBy(classifier, HashMap::new, downstream);
    }

    // 第三个
    public static <T, K, D, A, M extends Map<K, D>>
    Collector<T, ?, M> groupingBy(Function<? super T, ? extends K> classifier,
                                  Supplier<M> mapFactory,
                                  Collector<? super T, A, D> downstream) {
        Supplier<A> downstreamSupplier = downstream.supplier();
        BiConsumer<A, ? super T> downstreamAccumulator = downstream.accumulator();
        BiConsumer<Map<K, A>, T> accumulator = (m, t) -> {
            K key = Objects.requireNonNull(classifier.apply(t), "element cannot be mapped to a null key");
            A container = m.computeIfAbsent(key, k -> downstreamSupplier.get());
            downstreamAccumulator.accept(container, t);
        };
        BinaryOperator<Map<K, A>> merger = Collectors.<K, A, Map<K, A>>mapMerger(downstream.combiner());
        @SuppressWarnings("unchecked")
        Supplier<Map<K, A>> mangledFactory = (Supplier<Map<K, A>>) mapFactory;

        if (downstream.characteristics().contains(Collector.Characteristics.IDENTITY_FINISH)) {
            return new CollectorImpl<>(mangledFactory, accumulator, merger, CH_ID);
        }
        else {
            @SuppressWarnings("unchecked")
            Function<A, A> downstreamFinisher = (Function<A, A>) downstream.finisher();
            Function<Map<K, A>, M> finisher = intermediate -> {
                intermediate.replaceAll((k, v) -> downstreamFinisher.apply(v));
                @SuppressWarnings("unchecked")
                M castResult = (M) intermediate;
                return castResult;
            };
            return new CollectorImpl<>(mangledFactory, accumulator, merger, finisher, CH_NOID);
        }
    }

可以看到第三个函数是主要的实现分组功能的,有参数:classifier(分组条件)、mapFactory(分组用的map),downstream。

所以可以直接使用含三个参数的函数,传入有顺序的Map,LinkedHashMap,这样分组之后的顺序就是原来List集合的数据。

groupingBy(User::getAge(), LinkedHashMap::new, Collectors.toList())

HaseMap是无序的Map,是根据key的hashcode进行hash,然后放入对应的地址。所以在按照一定顺序put进HashMap之后,遍历HashMap的顺序跟put的顺序不同。

  • 3
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java 8中,使用Stream的Collectors.groupingBy方法可以对数据进行分组操作。但是需要注意的是,如果分组的字段数据存在丢失的情况,可能是由于以下原因导致的: 1. 分组字段的数据类型不一致:如果分组字段的数据类型不一致,可能会导致分组数据丢失。例如,如果分组字段是一个对象的属性,而该属性的数据类型在不同的对象中不一致,那么在分组可能会导致某些数据被丢弃。 2. 分组字段的hashCode和equals方法未正确重写:在进行分组操作,需要使用分组字段的hashCode和equals方法来确定分组的依据。如果这两个方法未正确重写,可能会导致分组数据丢失。 为了避免数据丢失的情况发生,可以采取以下措施: 1. 确保分组字段的数据类型一致:在进行分组操作之前,可以先对分组字段的数据类型进行统一,保它们具有相同的数据类型。 2. 重写分组字段的hashCode和equals方法:如果分组字段是一个自定义对象的属性,需要确保该属性的hashCode和equals方法已正确重写,以确保分组操作的准确性。 下面是一个示例代码,演示了如何使用Java 8的Stream和Collectors.groupingBy方法进行分组操作: ```java import java.util.Arrays; import java.util.List; import java.util.Map; import java.util.stream.Collectors; public class GroupingByDemo { public static void main(String[] args) { List<String> names = Arrays.asList("Alice", "Bob", "Charlie", "David", "Eva"); // 按照名字的首字母进行分组 Map<Character, List<String>> groups = names.stream() .collect(Collectors.groupingBy(name -> name.charAt(0))); // 输出分组结果 groups.forEach((key, value) -> System.out.println(key + ": " + value)); } } ``` 运行以上代码,将会按照名字的首字母进行分组,并输出分组结果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值