首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >是否可以使用.flatMap()或.collect()均匀地多个集合

是否可以使用.flatMap()或.collect()均匀地多个集合
EN

Stack Overflow用户
提问于 2016-06-08 12:36:45
回答 3查看 2.7K关注 0票数 9

例如,有集合[1,2,3,4,5][6,7,8][9,0]。通过Java8StreamAPI避免循环与迭代器交织这些集合以获得以下结果的任何方法-- [1,6,9,2,7,0,3,8,4,5]

EN

回答 3

Stack Overflow用户

回答已采纳

发布于 2016-06-08 14:30:29

我不确定用Stream是否有一种更简单的方法,但是您可以使用所有列表的索引上的流来处理这个问题:

代码语言:javascript
复制
static <T> List<T> interleave(List<List<T>> lists) {
    int maxSize = lists.stream().mapToInt(List::size).max().orElse(0);
    return IntStream.range(0, maxSize)
                    .boxed()
                    .flatMap(i -> lists.stream().filter(l -> i < l.size()).map(l -> l.get(i)))
                    .collect(Collectors.toList());
}

这将获得给定列表中最大列表的大小。对于每个索引,它将其与该索引中每个列表的元素形成的流(如果存在该元素)进行平面映射。

然后你就可以用它

代码语言:javascript
复制
public static void main(String[] args) {
    List<Integer> list1 = Arrays.asList(1,2,3,4,5);
    List<Integer> list2 = Arrays.asList(6,7,8);
    List<Integer> list3 = Arrays.asList(9,0);
    System.out.println(interleave(Arrays.asList(list1, list2, list3))); // [1, 6, 9, 2, 7, 0, 3, 8, 4, 5]
}

使用质子包库,您可以使用interleave方法执行以下操作:

代码语言:javascript
复制
List<Stream<Integer>> lists = Arrays.asList(list1.stream(), list2.stream(), list3.stream());
List<Integer> result = StreamUtils.interleave(Selectors.roundRobin(), lists).collect(Collectors.toList());
System.out.println(result);
票数 9
EN

Stack Overflow用户

发布于 2016-08-19 09:37:13

但不是一个很好的解决方案。只是一次尝试

代码语言:javascript
复制
    List<String> listA = Arrays.asList("a1","a2","a3","a4");
    List<String> listB = Arrays.asList("b1","b2","b3","b4");
    List<String> listC = Arrays.asList("c1","c2","c3","c4");
    List<String> combined = Stream.of(listA, listB, listC).flatMap(Collection::stream).collect(Collectors.toList());
    AtomicInteger index = new AtomicInteger();
    List<String> list1 = combined
            .stream().filter(x -> index.incrementAndGet() % 4 ==1).collect(Collectors.toList());
    index.set(0);
    List<String> list2 = combined
    .stream().filter(x -> index.incrementAndGet() % 4 ==2).collect(Collectors.toList());
    index.set(0);
    List<String> list3 = combined
    .stream().filter(x -> index.incrementAndGet() % 4 ==3).collect(Collectors.toList());
    index.set(0);
    List<String> list0 = combined
            .stream().filter(x -> index.incrementAndGet() % 4 ==0).collect(Collectors.toList());
    List<String> desiredOutput = Stream.of(list1, list2, list3,list0).flatMap(Collection::stream).collect(Collectors.toList());
    System.out.println(String.join(",", desiredOutput));
票数 0
EN

Stack Overflow用户

发布于 2016-08-21 15:09:44

我喜欢Tunaki的回答,并且一直在研究这个问题:

代码语言:javascript
复制
static <T> List<T> interleave(List<? extends Collection<T>> lists) {
    int maxSize = lists.stream().mapToInt(Collection::size).max().orElse(0);
    List<Iterator<T>> iterators = lists.stream().map(l -> l.stream().iterator()).collect(Collectors.toList());
    return IntStream.range(0, maxSize)
            .boxed()
            .flatMap(i -> iterators.stream().filter(it -> it.hasNext()).map(it -> it.next()))
            .collect(Collectors.toList());
}

我想看看我们是否也能把最初的列表作为流来处理,结果证明我们可以。作为一个副作用,我们避免了列表中的get(i) (如果这真的是一个很长的LinkedList,我们希望避免它)。我没有找到一种方法,不知道的最大长度的原始名单。

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/37702703

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档