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

问题
一般情况下,我们拿到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的顺序不同。

你可能感兴趣的:(java,开发语言)