浅谈java8streamflatMap流的扁平化操作
概念:
Steam 是Java8 提出的⼀个新概念,不是输⼊输出的 Stream 流,⽽是⼀种⽤函数式编程⽅式在集合类上进⾏复杂操作的⼯具。简⽽⾔之,是以内部迭代的⽅式处理集合数据的操作,内部迭代可以将更多的控制权交给集合类。Stream 和 Iterator 的功能类似,只是 Iterator 是以外部迭代的形式处理集合数据的操作。
在Java8以前,对集合的操作需要写出处理的过程,如在集合中筛选出满⾜条件的数据,需要⼀⼀遍历集合中的每个元素,再把每个元素逐⼀判断是否满⾜条件,最后将满⾜条件的元素保存返回。⽽Stream 对集合筛选的操作提供了⼀种更为便捷的操作,只需将实现函数接⼝的筛选条件作为参数传递进来,Stream会⾃⾏操作并将合适的元素同样以stream 的⽅式返回,最后进⾏接收即可。
2种操作:
1.intermediate 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()).List());
for (int i = 0, length = list.size(); i < length; i++) {
System.out.(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()
.List());
System.out.println(res);
java stream
//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).List());
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})).List());
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})).List());
System.out.println(res3.size());
}
}
控制台输出结果:
补充知识:Java 之 Stream流中map和flatMap的区别
我们先来看 map。如果你熟悉 scala 这类函数式语⾔,对这个⽅法应该很了解,它的作⽤就是把 input Stream 的每⼀个元素,映射成 output Stream 的另外⼀个元素。
转换⼤写
List<String> output = wordList.stream().
map(String::toUpperCase).
List());
这段代码把所有的单词转换为⼤写。
平⽅数
List<Integer> nums = Arrays.asList(1, 2, 3, 4);
List<Integer> squareNums = nums.stream().
map(n -> n * n).
List());
这段代码⽣成⼀个整数 list 的平⽅数 {1, 4, 9, 16}。
从上⾯例⼦可以看出,map ⽣成的是个 1:1 映射,每个输⼊元素,都按照规则转换成为另外⼀个元素。还有⼀些场景,是⼀对多映射关系的,这时需要 flatMap。
⼀对多
Stream<List<Integer>> inputStream = Stream.of(
Arrays.asList(1),
Arrays.asList(2, 3),
Arrays.asList(4, 5, 6)
);
Stream<Integer> outputStream = inputStream.
flatMap((childList) -> childList.stream());
flatMap 把 inpuStream 中的层级结构扁平化,就是将最底层元素抽出来放到⼀起,最终 output 的新 Stream ⾥⾯已经没有 List 了,都是直接的数字。
以上这篇浅谈java8 stream flatMap流的扁平化操作就是⼩编分享给⼤家的全部内容了,希望能给⼤家⼀个参考,也希望⼤家多多⽀持。

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。