1.引言

  如果你试图对流操作中的流水线进行调试, 了解stream流水线每个操作之前和操作之后的中间值, 该如何去做?

  首先我们看一个例子, 使用forEach将流操作的结果打印出来.

 1 /**
2 * @author lyh
3 * @version v-1.0.0
4 * @since 2021/5/28
5 */
6 public class PeekTestOne {
7 public static void main(String[] args) {
8 List<Integer> list = Arrays.asList(4, 7, 9, 11, 12);
9 list.stream()
10 .map(x -> x + 2)
11 .filter(x -> x % 2 != 0)
12 .limit(2)
13 .forEach(System.out::println);
14 }
15 }
16 输出结果如下:
17 9
18 11

  可以很明显的看出, 一旦调用了forEach操作, 整个流就会恢复运行.并不能很好的帮助我们了解Stream流水线中的每个操作(如:map,filter,limit等)产生的输出.

  再来看一个例子

 1 /**
2 * @author lyh
3 * @version v-1.0.0
4 * @since 2021/5/28
5 */
6 public class PeekTestTwo {
7 public static void main(String[] args) {
8 Stream<Integer> stream = Arrays.asList(4, 7, 9, 11, 12).stream();
9 stream.peek(System.out::println);
10
11 }
12 }
13 这段代码是想打印stream中的值,却没有任何输出.

2.中间操作和终止操作

  中间操作是流水线中的数据进行加工的, 它是一个懒操作, 并不会马上执行, 需要等待有终止操作的时候才会执行.

  终止操作是Stream的启动操作, 当有终止操作的时候, Stream才会真正的开始执行.

  因此, 这里可以解释上面的peek操作是一个中间操作, 所以没有任何输出.

3.使用peek进行debug操作

peek的设计初衷就是在流的每个元素恢复运行之前的时候插入一个执行操作. 它不想forEach那样恢复整个流的运行操作. 而是在一个元素上完成操作之后, 它只会将操作顺承到流水线的下一个操作. 它能够将中间变量的值输出到日志. 有效的帮助我们了解流水线的每一步操作的输出值.如下图:

 1 /**
2 * @author lyh
3 * @version v-1.0.0
4 * @since 2021/5/28
5 */
6 public class PeekTestThree {
7 public static void main(String[] args) {
8 List<Integer> list = Arrays.asList(4, 7, 9, 11, 12);
9 list.stream()
10 .peek(x -> System.out.println("stream: " + x))
11 .map(x -> x + 2)
12 .peek(x -> System.out.println("map: " + x))
13 .filter(x -> x % 2 != 0)
14 .peek(x -> System.out.println("filter: " + x))
15 .limit(2)
16 .peek(x -> System.out.println("limit: " + x))
17 .collect(toList());
18 }
19 }
20 输出结果如下:
21 stream: 4
22 map: 6
23 stream: 7
24 map: 9
25 filter: 9
26 limit: 9
27 stream: 9
28 map: 11
29 filter: 11
30 limit: 11
31
32 Process finished with exit code 0

4.peek和map的区别

  使用peek操作流,流中的元素没有改变。

 1 /**
2 * @author lyh
3 * @version v-1.0.0
4 * @since 2021/5/28
5 */
6 public class PeekAndMapTestOne {
7 public static void main(String[] args) {
8 Arrays.asList("a","b")
9 .stream()
10 .peek(x -> x.toUpperCase())
11 .forEach(System.out::println);
12 }
13 }
14 输出:
15 a
16 b
17
18 Process finished with exit code 0

  使用map操作流,流中的元素有改变。

 1 /**
2 * @author lyh
3 * @version v-1.0.0
4 * @since 2021/5/28
5 */
6 public class PeekAndMapTestTwo {
7 public static void main(String[] args) {
8 Arrays.asList("a","b")
9 .stream()
10 .map(x -> x.toUpperCase())
11 .forEach(System.out::println);
12 }
13 }
14 输出:
15 A
16 B
17
18 Process finished with exit code 0

  可以通过上面两个例子看出,map操作是对元素进行了转换。

  注意:peek对一个对象进行操作的时候,对象不变,但是可以改变对象里面的值.如下:

 1 /**
2 * @author lyh
3 * @version v-1.0.0
4 * @since 2021/5/28
5 */
6 @Getter
7 @Setter
8 @AllArgsConstructor
9 @ToString
10 public class Person {
11
12 private String id;
13 private String name;
14
15 }
16 ----------------------------------------------------------------------------
17 /**
18 * @author lyh
19 * @version v-1.0.0
20 * @since 2021/5/28
21 */
22 public class PeekAndMapTestThree {
23 public static void main(String[] args) {
24 Arrays.asList(new Person("001","zs"),new Person("002","ls"))
25 .stream().peek(p -> p.setId("000")).forEach(System.out::println);
26 }
27 }
28 输出:
29 Person(id=000, name=zs)
30 Person(id=000, name=ls)
31
32 Process finished with exit code 0

  peek的定义

1 Stream<T> peek(Consumer<? super T> action);

  peek方法接收一个Consumer的入参. 了解λ表达式的应该明白 Consumer的实现类应该只有一个方法,该方法返回类型为void. 它只是对Stream中的元素进行某些操作,但是操作之后的数据并不返回到Stream中,所以Stream中的元素还是原来的元素.

  map的定义

1 <R> Stream<R> map(Function<? super T, ? extends R> mapper);

  map方法接收一个Function作为入参. Function是有返回值的, 这就表示map对Stream中的元素的操作结果都会返回到Stream中去.

​   觉得此文不错, 点赞+转发+关注, 本人非常感谢!

Stream中的Peek操作的更多相关文章

  1. Java 8 Stream Api 中的 peek 操作

    1. 前言 我在Java8 Stream API 详细使用指南[1] 中讲述了 [Java 8 Stream API]( "Java 8 Stream API") 中 map 操作 ...

  2. Stream流的这些操作,你得知道,对你工作有很大帮助

    Stream流 Stream(流)是一个来自数据源的元素队列并支持聚合操作: 元素是特定类型的对象,形成一个队列. Java中的Stream并不会存储元素,而 是按需计算. 数据源 流的来源. 可以是 ...

  3. java 8 Stream中操作类型和peek的使用

    目录 简介 中间操作和终止操作 peek 结论 java 8 Stream中操作类型和peek的使用 简介 java 8 stream作为流式操作有两种操作类型,中间操作和终止操作.这两种有什么区别呢 ...

  4. Java8中的流操作-基本使用&性能测试

    为获得更好的阅读体验,请访问原文:传送门 一.流(Stream)简介 流是 Java8 中 API 的新成员,它允许你以声明式的方式处理数据集合(通过查询语句来表达,而不是临时编写一个实现).这有点儿 ...

  5. Java8 Stream流API常用操作

    Java版本现在已经发布到JDK13了,目前公司还是用的JDK8,还是有必要了解一些JDK8的新特性的,例如优雅判空的Optional类,操作集合的Stream流,函数式编程等等;这里就按操作例举一些 ...

  6. 【JDK8】Java8 Stream流API常用操作

    Java版本现在已经发布到JDK13了,目前公司还是用的JDK8,还是有必要了解一些JDK8的新特性的,例如优雅判空的Optional类,操作集合的Stream流,函数式编程等等;这里就按操作例举一些 ...

  7. JAVA8之lambda表达式具体解释,及stream中的lambda使用

    前言: 本人也是学习lambda不久,可能有些地方描写叙述有误,还请大家谅解及指正! lambda表达式具体解释 一.问题 1.什么是lambda表达式? 2.lambda表达式用来干什么的? 3.l ...

  8. java中的集合操作类(未完待续)

    申明: 实习生的肤浅理解,如发现有错误之处.还望大牛们多多指点 废话 事实上我写java的后台操作,我每次都会遇到一条语句:List<XXXXX> list = new ArrayList ...

  9. loadrunner中文件的操作

    loadrunner中文件的操作 我们可以使用fopen().fscanf().fprintf().fclose()函数进行文件操作,但是因为LoadRunner不支持FILE数据类型,所以我们需要做 ...

随机推荐

  1. 后端Spring Boot+前端Android交互+MySQL增删查改(Java+Kotlin实现)

    1 前言&概述 这篇文章是基于这篇文章的更新,主要是更新了一些技术栈以及开发工具的版本,还有修复了一些Bug. 本文是SpringBoot+Android+MySQL的增删查改的简单实现,用到 ...

  2. 6. Mybatis resultMap

    resultMap 元素是MyBatis中最重要最强大的元素.它就是让你远离90%的需要从结果集中取出数据的JDBC代码的那东西,而且在一些情形下允许你做一些JDBC不支持的事情.事实上,编写相似于对 ...

  3. 1087 All Roads Lead to Rome

    Indeed there are many different tourist routes from our city to Rome. You are supposed to find your ...

  4. hdu4862 费用流(不错)

    题意:       给你一个矩阵,你最多可以选择k条路线,k条路线的起点随意,每次行走的距离随意,但是只能往右或者下走,走过的点不能再走,而且每一步如果a->b,如果a和b的权值s相等那么就可以 ...

  5. POJ1466 最大点权独立集

    题意:       给你n个人,再给你每个人都喜欢哪些人,让你找到一个最大的集合数,要求这个集合里面任意两个人都不喜欢彼此. 思路:       直接就是在问最大点权独立集元素个数,没啥解释的一遍二分 ...

  6. 路由器逆向分析------在Linux上安装IDA Pro

    本文博客地址:http://blog.csdn.net/qq1084283172/article/details/69665905 01.在Linux系统上安装Linux版本的IDA Pro Linu ...

  7. Python中的socket网络模块

    目录 Socket 服务端(server.py) 客户端(client.py) socket中的一些常用方法 Socket 对象(内建)方法 Python Internet 模块 Python3 提供 ...

  8. LA3027简单带权并查集

    题意:       有n个点,一开始大家都是独立的点,然后给出一些关系,a,b表示a是b的父亲节点,距离是abs(a-b)%1000,然后有一些询问,每次询问一个节点a到父亲节点的距离是多少? 思路: ...

  9. CreateThread 线程操作与 _beginthreadex 线程安全(Windows核心编程)

    0x01 线程的创建 线程不同于进程,Windows 中的进程是拥有 '惰性' 的,本身并不执行任何代码,而执行代码的任务转交给主线程,列如使用 CreateProcess 创建一个进程打开 Cmd ...

  10. (3) arm 指令

    跳转指令1.1 B 跳转指令 B{cond} label 说明:如果条件cond满足,arm处理器将立即跳转到label指定的地址处继续执行.1.2 BL 带链接的跳转指令 BL{cond} labe ...