java8 stream flatMap流的扁平化操作

概念:Steam 是Java8 提出的一个新概念,不是输入输出的 Stream 流,而是一种用函数式编程方式在集合类上进行复杂操作的工具。简而言之,是以内部迭代的方式处理集合数据的操作,内部迭代可以将更多的控制权交给集合类。Stream 和 Iterator 的功能类似,只是 Iterator 是以外部迭代的形式处理集合数据的操作。

在Java8以前,对集合的操作需要写出处理的过程,如在集合中筛选出满足条件的数据,需要一 一遍历集合中的每个元素,再把每个元素逐一判断是否满足条件,最后将满足条件的元素保存返回。而Stream 对集合筛选的操作提供了一种更为便捷的操作,只需将实现函数接口的筛选条件作为参数传递进来,Stream会自行操作并将合适的元素同样以stream 的方式返回,最后进行接收即可。

2种操作:

1.intermediate  operation 中间操作:中间操作的结果是刻画、描述了一个Stream,并没有产生一个新集合,这种操作也叫做惰性求值方法。

2.terminal operation 终止操作:最终会从Stream中得到值。

如何区分这2种操作呢?可以根据操作的返回值类型判断,如果返回值是Stream,则该操作是中间操作,如果返回值是其他值或者为空,则该操作是终止操作。
 

flatMap 中间操作:

可用 Stream 替换值,并将多个 Stream 流合并成一个 Stream 流。

将含有一串数字的两个流合并为一个流,

	@Test
	public void flapMapTest() {
		List<Integer> list = (List<Integer>) Stream.of(Arrays.asList(1, 2, 3, 4, 5, 6), Arrays.asList(8, 9, 10, 11, 12))
			.flatMap(test -> test.stream()).collect(Collectors.toList());
 
		for (int i = 0, length = list.size(); i < length; i++) {
			System.out.println(list.get(i));
		}
 
	}

flatMap的用法和含义住要通过一个案例来讲解,

案例:对给定单词列表 ["Hello","World"],你想返回列表["H","e","l","o","W","r","d"]

第一种方式


String[] words = new String[]{"Hello","World"};

List<String[]> a = Arrays.stream(words)

.map(word -> word.split(""))

.distinct()

.collect(toList());

a.forEach(System.out::print);

       代码输出为:[Ljava.lang.String;@12edcd21[Ljava.lang.String;@34c45dca 

                            (返回一个包含两个String[]的list)

        这个实现方式是由问题的,传递给map方法的lambda为每个单词生成了一个String[](String列表)。因此,map返回的流实际上是Stream<String[]> 类型的。你真正想要的是用Stream<String>来表示一个字符串。

        下方图是上方代码stream的运行流程

 

第二种方式:flatMap(对流扁平化处理)

String[] words = new String[]{"Hello","World"};

List<String> a = Arrays.stream(words)

.map(word -> word.split(""))

.flatMap(Arrays::stream)

.distinct()

.collect(toList());

a.forEach(System.out::print);

结果输出:HeloWrd

        使用flatMap方法的效果是,各个数组并不是分别映射一个流,而是映射成流的内容,所有使用map(Array::stream)时生成的单个流被合并起来,即扁平化为一个流。

    下图是运用flatMap的stream运行流程,

    示例:


import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;
 
 
public class FlatMap {
    public static void main(String[] args) {
        //扁平化流
        //找出数组中唯一的字符
        String[] strArray = {"hello", "world"};
 
        //具体实现
        List<String> res = Arrays.stream(strArray)
                .map(w -> w.split(""))
                .flatMap(Arrays::stream)
                .distinct()
                .collect(Collectors.toList());
        System.out.println(res);
 
        //TODO 案例
        System.out.println("--------------------------------");
        //Demo1:给定数组,返回数组平方和(直接使用映射)
        //[1,2,3,4]=>[1,4,9,16]
        Integer[] nums1 = {1, 2, 3, 4};
        List<Integer> nums1List = Arrays.asList(nums1);
        List<Integer> res1 = nums1List.stream().map(i -> i * i).collect(Collectors.toList());
        System.out.println(res1);
 
        System.out.println("--------------------------------");
        //Demo2:给定两数组,返回数组对
        //[1,2,3],[3,4]=>[1,3],[1,4],[2,3],[2,4],[3,3],[3,4]
        Integer[] nums2 = {1, 2, 3};
        Integer[] nums3 = {3, 4};
        List<Integer> nums2List = Arrays.asList(nums2);
        List<Integer> nums3List = Arrays.asList(nums3);
 
        //使用2个map嵌套过滤
        List<int[]> res2 = nums2List.stream().flatMap(i -> nums3List.stream().map(j -> new int[]{i, j})).collect(Collectors.toList());
        System.out.println(res2.size());
 
        System.out.println("--------------------------------");
        //Demo3:针对Demo2和Demo1组合返回总和能被3整除的数对
        //(2,4)和(3,3)是满足条件的
        List<int[]> res3 = nums2List.stream().flatMap(i -> nums3List.stream().filter(j -> (i + j) % 3 == 0).map(j -> new int[]{i, j})).collect(Collectors.toList());
        System.out.println(res3.size());
 
 
    }
}

控制台输出结果:

  • 6
    点赞
  • 37
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
Java Stream API是Java 8引入的一种新的API,它能够将集合或数组等数据源转化为一个,从而支持一些强大的、高效的函数式编程操作,简化了我们对集合数据进行处理的代码。其中,`flatmap()`操作就是Java Stream API中的一个反扁平操作扁平化操作指的是将嵌套的数据结构展开,并合并成单一的数据结构。比如说,如果我们有一个嵌套的列表,我们可以使用`flatmap()`操作把它展开成一个扁平的列表。而反扁平化操作则是把数据结构重新组合,由扁平的对象重组成嵌套的数据结构。 `FlatMap()`方法将每个元素转换为一个Stream对象,然后将所有的Stream对象合并为一个Stream对象进行处理。与此相反,`flatMap()`操作将多个转化为一个,并将其中的元素合并处理。在实际应用中,`flatMap()`方法常被用于将多层结构的数据进行平铺操作。 举个例子,假设有一个List<List<Integer>>的集合,我们要将它转换为一个List<Integer>的集合,就可以使用`flatMap()`操作,代码如下: ```java List<List<Integer>> list = Arrays.asList( Arrays.asList(1, 2, 3), Arrays.asList(4, 5, 6), Arrays.asList(7, 8, 9) ); List<Integer> result = list.stream() .flatMap(Collection::stream) .collect(Collectors.toList()); System.out.println(result); ``` 输出结果为:[1, 2, 3, 4, 5, 6, 7, 8, 9] 可以看到,`flatMap()`操作将多个Stream对象中的元素取出,并将它们合并成为一个Stream对象,最终返回一个扁平的List<Integer>。相反,反扁平化操作 `unflatMap()`就是把一个扁平化的列表转换回原始的嵌套列表结构。 总之,Java Stream API中的`flatMap()`操作是非常强大、灵活的,能够处理许多嵌套的结构。它是函数式编程强大的工具之一,可以让我们更加方便地处理集合数据,提高代码的可读性和效率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ZhaoYingChao88

整理不易,还请多多支持,谢谢

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值