Java8使用Stream的缺点是调试困难?教你一招你就不会这么认为了

Posted Hollis Chuang

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java8使用Stream的缺点是调试困难?教你一招你就不会这么认为了相关的知识,希望对你有一定的参考价值。

今天给大家分享如何在 IntellJ IDEA 中调试 Java8 带来牛逼哄哄的的新特性 Stream。

写在前

Java 8 API添加了一个新的抽象称为流Stream,可以让你以一种声明的方式处理集合数据。Stream API可以极大提高Java程序员的生产力,让我们写出高效率、干净、简洁的代码。

这种风格将要处理的元素集合看作一种流, 流在管道中传输, 并且可以在管道的节点上进行处理, 比如筛选, 排序,聚合等。元素流在管道中经过中间操作(intermediate operation)的处理,最后由最终操作(terminal operation)得到前面处理的结果。

+--------------------+       +------+   +------+   +---+   +-------+
| stream of elements +-----> |filter+-> |sorted+-> |map+-> |collect|
+--------------------+       +------+   +------+   +---+   +-------+

以上的流程转换为 Java 代码为:

List<Integer> transactionsIds = 
widgets.stream()
             .filter(b -> b.getColor() == RED)
             .sorted((x,y) -> x.getWeight() - y.getWeight())
             .mapToInt(Widget::getWeight)
             .sum();

Java 代码这样写,表达的意思也很明确,书写起来不要太流畅哦😯~

反正自从我们项目组升级到 Java8 后,一般涉及到集合遍历、元素转换、过滤、排序、统计,我反手就是一个 Stream。身边同事基本上也都是这么用的,因为书写起来实在太流畅了,feel倍儿爽~

痛点

之前我面试阿里的时候,二面的面试官就问到 Java8 都有哪些新特性呢?其中我就提到了 Java8 带来的 Stream,然后他就问缺点是什么?我回答写的代码难以调试,因为不像for循环那样可以每一行打断点调试了。

Java7中我们计算空字符串的数量可以使用如下代码:

// 计算空字符串
List<String> strings = Arrays.asList("abc", "", "bc", "efg", "abcd", "", "jkl");
System.out.println("列表: " + strings);
int count = 0;
for (String string : strings) {
    if (string.isEmpty()) {
        count++;
    }
}
return count;

调试的时候直接在循环中打上断点,以Debug方式运行,就可以跟踪代码的执行流程了:

来,我们再看下Java8中的实现代码:

System.out.println("使用 Java 8: ");
count = strings.stream().filter(string -> string.isEmpty()).count();
System.out.println("空字符串数量为: " + count);

就一行代码,我们需要怎么打断点调试呢?

如上图所示,在 Stream 代码的这一行打断点,如果你选择 Line ,那么就无法调试,跟踪不到Stream在管道中传输以及在管道的节点上进行的filter处理动作。

选择断点加在 lambda表达式上,然后使用单步调试才可以进入,不得不说 IDEA 是真的强👍🏻。我印象中之前使用2019版本打断点是没有出现这个提示的,反正2020以上的版本肯定都有了,打断点的时候就会提示选择。

虽然可以调试,问题是可以解决了,不过还不够强,接下来大家别眨眼睛,一个更强大、更直观的视图,帮助我们一眼就能看出 Stream 的处理过程。

可视化追踪 Stream 链

步骤还是上面的一样,打断点,以Debug的方式运行程序,区别在于打断点时无需选择是行端点还是lambda表达式上面,随便选择只要打上断点即可:

点击图中按钮,就会自动打开一个Stream处理流程的视图,整个处理过程变得一目了然,视图分为三分部,左边是初始集合的数据,中间是Stream处理过滤后的数据,右边是最终操作得到的处理结果。

这样Java8 Stream 相关的API( 筛选, 排序,聚合)操作就都可以可视化的展示出来了,调试时非常的方便,排查问题岂不是一眼就看到问题所在了。你说这个玩意香不香呢😋?

有道无术,术可成;有术无道,止于术

欢迎大家关注Java之道公众号

好文章,我在看❤️

以上是关于Java8使用Stream的缺点是调试困难?教你一招你就不会这么认为了的主要内容,如果未能解决你的问题,请参考以下文章

教你一招!使用IDEA远程调试Tomcat...

JAVA8之lambda表达式详解,及stream中的lambda使用

我们一起来学java8的lambda表达式Stream

教你一步步封装vue的toast和loading组件

如何使用 lambda 表达式调试 stream().map(...)?

IntelliJ IDEA中如何优雅的调试Java Stream操作