今天给大家分享如何在 IntellJ IDEA 中调试 Java8 带来牛逼哄哄的的新特性 Stream。
写在前面
Java8 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( 筛选, 排序,聚合)操作就都可以可视化的展示出来了,调试时非常的方便,排查问题岂不是一眼就看到问题所在了。你说这个玩意香不香呢😋?