上一次【http://www.cnblogs.com/webor2006/p/8297603.html】在最后用stream.iterate()生成了6个奇数,接着基于它来实现如下需求:找出该流中大于2的元素,然后再将每个元素乘以2,然后忽略掉流中的前两个元素,然后再取流中的前两个元素,最后求出流中元素的总和。那具体如何来实现呢?下面一个个条件来实现:

①、找出该流中大于2的元素。

很显然这是一个过滤操作,所以可以使用stream.fitler(),如下:

②、再将每个元素乘以2。

这个当然就是一种元素映射啦,所以可以使用stream.map(),如下:

而它接收的ToIntFunction的接口原型如下:

其具体代码如下:

那有个细节需要思考一下:

其实是为了避免自动装箱、自动拆箱, 为什么?对于map()这个方法要求的Function接口的返回结果必须是Integer类型的,而期望的最终结果应该是int类型的,所以用map()的话则就会自动装箱与自动拆箱,对于它是有一点点性能上的损耗的,可见Java8是极力想办法避勉这种细微的性能问题,因此这些特定类型的方法就应运而生,像mapToInt()中要求的ToIntFunction接口其返回结果就是直接的int类型,所以就不存在自动装箱与自动拆箱的问题,体会一下。

③、然后忽略掉流中的前两个元素。

怎么忽略呢?当然在流中已经提供了现成的方法啦,如下:

④、再取流中的前两个元素。

这时limit()就可以再次发挥作用,如下:

⑤、最后求出流中元素的总和。

直接可以使用stream.sum()方法,如下:

编译运行:

是不是32呢?咱们手动算一下:生成的6个奇数序列为:1, 3, 5, 7, 9, 11,接着从这6个中过滤出大于2的数为:3, 5, 7, 9, 11,然后再对它们进行乘2如下:6,10,14,18,22,然后忽略掉前两个数此时变为:14, 18, 22,再取出前两个元素如下:14, 18,最后求总和为:14 + 18 = 32,嗯~~没问题。

对于这么多的条件采用stream来实现是多么的简便,而且代码的可读性也比较好,试想一下如果采用传统的做法是不是大量的循环临时变量充斥其中,所以说Java8中Lambda表达式、函数式接口、Stream所带来的是突破性的改变。

接下来对于上面的需求再修改一下,将最后的元素求和改为最元素中最小的那个并打印出来 ,那如何搞呢?Stream中也提供有现成的方法,如下:

编译运行:

Optional咱们之前已经学习过了,对于OptionalInt那当然很好理解啦,里面的元素就是int类型,而非是一个泛型了,如下:

另外看一个方法定义的细节:

这是为什么呢?其实是返回int还是OptionalInt根本原因就是其取出来的值有没有可能为空,对于sum()方法而言,如果Stream中的元素为空,那最终直接显示0就好了,如下:

而如果换成min()或max(),这时会得到一个空的Stream,如下:

这时后如果在min()后面再去调用就会出问题了,如下:

所以因为min()返回的是OptionalInt类型,所以得用标准的用法,如下:

如果将过滤条件改成正常滴,如下:

目前stream()已经提供了求最小值、最大值、总和的方法,那如果想一次性将这三个需求全部求解出来,那怎么办呢?当然第一时间能想到的就是调用三次不同的方法不就可以解决了,但是这种方法显示是比较笨重的,其实Java8中还有更好的方法,如下:

其中看一下IntSummaryStatistics类的介绍:

其中针对咱们写的这个流操作的代码再来回顾一下它的分类:

接下来写一段如下代码,会有一些我们意想之外事情发生,如下:

上面由于调用的两个都是中间操作所以都会返回stream,那打印结果是啥呢?下面运行一下:

其重点看一下抛出的异常:流已经被操作了或者已经被关闭了,为什么?其实Java8中的流跟以前io中的流的概念其实是差不多的,意思是流如果被使用过了就不能再次使用了或者说如果流被关闭了也不能再次使用了,而对于咱们报错的这句代码显然流是木有被关闭的,如下:

那证明是该流已经被使用过了,再调用它的distinct()方法则就抛异常了,分析一下是不是这个原因:

那如何解决这个异常了,需对代码进行稍加修改,如下:

但是!!这里不是推荐的写法,推荐就是用链式的方式来写,而不要用临时的变量去接收再继续写,需要注意。

接下来再来探究另外一个问题就是关于stream中的中间操作与终止操作本质上的区别,举一个如下例子:将stream中的元素的首字母变成大写之后再将其输出,如下:

接下来改造一下代码,加入调试打印语句用来观察现象,如下:

编译运行:

如预期没有任何打印,此时如果再加上终止操作forEach(),

这也就是之前一直在说的关于stream包含两种操作:第一种是中间操作,第二种是终止操作,而对于流中所有的中间操作都是lazy的,也就是惰性求值的,如果没有遇到终止操作的时候它是不会执行的,只有当遇到了终止操作之后,而中间的若干个中间操作才会一并执行,这是需要谨记滴。

另上再回过头来看一下之前写的如下代码:

实际上不是咱们表面的想象的那样,要是性能低下Java8也不可能引进流的概念啦,实际上最终只会循环一次,为什么呢?可以这么简单的理解,以于这些中间操作的行为都是存放于Stream中的一个容器当中,一旦遇到终止操作时,则会将这些行为按照咱们写的顺序逐个的应用到集合当中的每一个元素上,所以说性能上不会有任何影响。

接下来继续看下面这个陷阱:

下面运行来论证一下我们的猜测:

这是为什么呢?

如何解决呢?当然就是先限定元素个数再来去重就成了,如下:

所以说对于stream里面的各种方法在实际编写代码时需要注意一下顺序,以勉掉到坑里面去。

java8学习之Stream陷阱剖析的更多相关文章

  1. java8学习之Stream实例剖析

    继续操练Stream,直接上代码: 而咱们要返回ArrayList,显示可以用构造引用来传递到里面,因为它刚好符合Supplier函数式接口的特性:不接收参数返回一个值,所以: 接下来试着将Strea ...

  2. java8学习之Stream分组与分区详解

    Stream应用: 继续举例来操练Stream,对于下面这两个集合: 需求是:将这两个集合组合起来,形成对各自人员打招呼的结果,输出的结果如: "Hi zhangsan".&quo ...

  3. java8学习之Stream介绍与操作方式详解

    关于默认方法[default method]的思考: 在上一次[http://www.cnblogs.com/webor2006/p/8259057.html]中对接口的默认方法进行了学习,那在Jav ...

  4. Java8学习(4)-Stream流

    Stream和Collection的区别是什么 流和集合的区别是什么? 粗略地说, 集合和流之间的差异就在于什么时候进行计算.集合是一个内存中的数据结构,它包含数据结构中目前所有的值--集合中的每个元 ...

  5. java8学习之Predicate深入剖析与函数式编程本质

    上次[http://www.cnblogs.com/webor2006/p/8214596.html]对Predicate函数接口进行了初步的学习,其中提到了在未来要学习的Stream中得到了大量的应 ...

  6. java8学习之Stream源码分析

    上一次已经将Collectors类中的各种系统收集器的源代码进行了完整的学习,而在之前咱们已经花了大量的篇幅对其Stream进行了详细的示例学习,如: 那接下来则通过源代码的角度来对Stream的运作 ...

  7. java8学习之Stream深度解析与源码实践

    继续对流进行学习,首先先说明一下流的特点: 1.Collection提供了新的stream()方法. 2.流不存储,通过管道的方式获取值. 3.本质是函数式的,对流的操作会生成一个结果,不过并不会修改 ...

  8. Java8学习笔记----Lambda表达式 (转)

    Java8学习笔记----Lambda表达式 天锦 2014-03-24 16:43:30 发表于:ATA之家       本文主要记录自己学习Java8的历程,方便大家一起探讨和自己的备忘.因为本人 ...

  9. 这可能是史上最好的 Java8 新特性 Stream 流教程

    本文翻译自 https://winterbe.com/posts/2014/07/31/java8-stream-tutorial-examples/ 作者: @Winterbe 欢迎关注个人微信公众 ...

随机推荐

  1. IDEA离线更新

    因为在hosts文件添加以下dns添加 0.0.0.0 account.jetbrains.com 0.0.0.0 www.jetbrains.com 导致IDEA自动更新失败.手工下载回更新文件IU ...

  2. java中instanceof的基本使用

    java中的instanceof运算符是用于判断对象是否是指定类或这个指定类的子类的一个实例,返回值是布尔类型. 语法: boolean result = object instanceof clas ...

  3. 如何把本地文件上传至github?

    (都说git好用,但我觉得git把我弄得像个git……在反反复复用git bash的命令行上传失败了N次之后,终于可以用命令行把文件上传到GitHub了 这中间,还要感谢网络上的各种git教程!!!) ...

  4. 从STL文件到网格拓扑

    原文链接 STL文件是什么 STL文件是网格文件的一种格式,分为二进制和文本两种类型.具体来讲,它定义了一群三角面片,比如下面是一个文本的STL示例: solid geometryplusplus f ...

  5. Star all over again.

    0x00前言 经过了一上午的折腾之后,博客的界面勉强可观,今天下午将之前的所有博客全部删除,重新开始写属于自己的博客,而不是只把它当作一个收藏夹,转载其他人的文章. 0x01近来感想 有感而发,随便写 ...

  6. kindeditor-4.1.7

    <script src="~/Scripts/jquery-1.10.2.min.js" type="text/javascript"></s ...

  7. 字符串转 Boolean 的正确方式

    String s1 = "false"; String s2 = "true"; String s3 = "fAlSe"; String s ...

  8. window启动目录

    启动文件目录 C:\Users\Administrator\AppData\Roaming\Microsoft\Windows\Start Menu\Programs\Startup

  9. java学习-3

    输入语句Scanner的使用方法 1.导包 import java.util.Scanner 2.创建 从键盘输入:Scanner sc = new Scanner(System.in); 3.使用 ...

  10. idea自动化部署插件 Alibaba Cloud Toolkit 使用记录

    官方安装文档和使用说明 https://help.aliyun.com/product/29966.html?spm=a2c4g.11186623.6.540.6efa6029JhlPfx 是什么? ...