转载请注明出处:https://zhuanlan.zhihu.com/p/20540202

Stream作为Java8的新特性之一,他与Java IO包中的InputStream和OutputStream完全不是一个概念。Java8中的Stream是对集合功能的一种增强,主要用于对集合对象进行各种非常便利高效的聚合和大批量数据的操作。结合Lambda表达式可以极大的提高开发效率和代码可读性。

假设我们需要把一个集合中的所有形状设置成红色,那么我们可以这样写

for 

如果使用Java8扩展后的集合框架则可以这样写:

shapes

第一种写法我们叫外部迭代,for-each调用shapes的iterator()依次遍历集合中的元素。这种外部迭代有一些问题:

  • for循环是串行的,而且必须按照集合中元素的顺序依次进行;
  • 集合框架无法对控制流进行优化,例如通过排序、并行、短路求值以及惰性求值改善性能。 > 上面这两个问题我们会在后面的文章中逐步解答。

第二种写法我们叫内部迭代,两段代码虽然看起来只是语法上的区别,但实际上他们内部的区别其实非常大。用户把对操作的控制权交还给类库,从而允许类库进行各种各样的优化(例如乱序执行、惰性求值和并行等等)。总的来说,内部迭代使得外部迭代中不可能实现的优化成为可能。

外部迭代同时承担了做什么(把形状设为红色)和怎么做(得到Iterator实例然后依次遍历),而内部迭代只负责做什么,而把怎么做留给类库。这样代码会变得更加清晰,而集合类库则可以在内部进行各种优化。

一、什么是Stream

Stream不是集合元素,它也不是数据结构、不能保存数据,它更像一个更高级的Interator。Stream提供了强大的数据集合操作功能,并被深入整合到现有的集合类和其它的JDK类型中。流的操作可以被组合成流水线(Pipeline)。拿前面的例子来说,如果我只想把蓝色改成红色:

shapes

在Collection上调用stream()会生成该集合元素的流,接下来filter()操作会产生只包含蓝色形状的流,最后,这些蓝色形状会被forEach操作设为红色。

如果我们想把蓝色的形状提取到新的List里,则可以:

List

collect()操作会把其接收的元素聚集到一起(这里是List),collect()方法的参数则被用来指定如何进行聚集操作。在这里我们使用toList()以把元素输出到List中。

如果每个形状都被保存在Box里,然后我们想知道哪个盒子至少包含一个蓝色形状,我们可以这么写:

Set

map()操作通过映射函数(这里的映射函数接收一个形状,然后返回包含它的盒子)对输入流里面的元素进行依次转换,然后产生新流。

如果我们需要得到蓝色物体的总重量,我们可以这样表达:

int 

二、Stream vs Collection

流(Stream)和集合(Collection)的区别:

  • Collection主要用来对元素进行管理和访问;
  • Stream并不支持对其元素进行直接操作和直接访问,而只支持通过声明式操作在其之上进行运算后得到结果;
  • Stream不存储值
  • 对Stream的操作会产生一个结果,但是Stream并不会改变数据源;
  • 大多数Stream的操作(filter,map,sort等)都是以惰性的方式实现的。这使得我们可以使用一次遍历完成整个流水线操作,并可以用短路操作提供更高效的实现。

三、惰性求值 vs 急性求值

filter()和map()这样的操作既可以被急性求值(以filter()为例,急性求值需要在方法返回前完成对所有元素的过滤),也可以被惰性求值(用Stream代表过滤结果,当且仅当需要时才进行过滤操作)在实际中进行惰性运算可以带来很多好处。比如说,如果我们进行惰性过滤,我们就可以把过滤和流水线里的其它操作混合在一起,从而不需要对数据进行多遍遍历。相类似的,如果我们在一个大型集合里搜索第一个满足某个条件的元素,我们可以在找到后直接停止,而不是继续处理整个集合。(这一点对无限数据源是很重要,惰性求值对于有限数据源起到的是优化作用,但对无限数据源起到的是决定作用,没有惰性求值,对无限数据源的操作将无法终止)

对于filter()和map()这样的操作,我们很自然的会把它当成是惰性求值操作,不过它们是否真的是惰性取决于它们的具体实现。另外,像sum()这样生成值的操作和forEach()这样产生副作用的操作都是天然急性求值,因为它们必须要产生具体的结果。

我们拿下面这段代码举例:

int 

这里的filter()和map()都是惰性的,这就意味着在调用sum()之前不会从数据源中提取任何元素。在sum()操作之后才会把filter()、map()和sum()放在对数据源一次遍历中。这样可以大大减少维持中间结果所带来的开销。

四、举个栗子

前面长篇大论的介绍概念实在太枯燥,为了方便大家理解我们用Streams API来实现一个具体的业务场景。

假设我们有一个房源库项目,这个房源库中有一系列的小区,每个小区都有小区名和房源列表,每套房子又有价格、面积等属性。现在我们需要筛选出含有100平米以上房源的小区,并按照小区名排序。

我们先来看看不用Streams API如何实现:

List

如果使用Streams API:

return 

如果你喜欢我的文章,就关注下我的公众号 BaronTalk知乎专栏 或者在 GitHub 上添个 Star 吧!

技术男的后花园​zhuanlan.zhihu.com

BaronZ88 - Overview​github.com

java8 lambda map排序_Java8新特性第3章(Stream API)相关推荐

  1. java8 lambda map排序_Android兼容Java 8语法特性的原理分析

    本文主要阐述了Lambda表达式及其底层实现(invokedynamic指令)的原理.Android第三方插件RetroLambda对其的支持过程.Android官方最新的dex编译器D8对其的编译支 ...

  2. sqlserver获取前一天的日期_Java8新特性时间日期库DateTime API及示例

    点击上方蓝字关注「程序新视界」 Java8新特性的功能已经更新了不少篇幅了,今天重点讲解时间日期库中DateTime相关处理.同样的,如果你现在依旧在项目中使用传统Date.Calendar和Simp ...

  3. java instant获取微秒转成日期格式_Java8新特性时间日期库DateTime API及示例

    点击上方蓝字关注「程序新视界」 Java8新特性的功能已经更新了不少篇幅了,今天重点讲解时间日期库中DateTime相关处理.同样的,如果你现在依旧在项目中使用传统Date.Calendar和Simp ...

  4. stream of java_java8新特性之强大的Stream API

    Stream API Stream是Java8中处理集合的关键抽象概念,它可以指定你希望对集合进行的操作,可以执行非常复杂的查找.过滤和映射数据等操作. 使用Stream API 对集合数据进行操作, ...

  5. jdk8銝要onematch_JDK 8 新特性之函数式编程 → Stream API

    开心一刻 今天和朋友们去K歌,看着这群年轻人一个个唱的贼嗨,不禁感慨道:年轻真好啊! 想到自己年轻的时候,那也是拿着麦克风不放的人 现在的我没那激情了,只喜欢坐在角落里,默默的听着他们唱,就连旁边的妹 ...

  6. java8 list map 排序_java8如何对list中的map元素根据多个key值进行排序

    使用JAVA自己的排序方法,有的时候是一个可行的选择. 先从简单的开始说起. 一.少数key的情况 有一个需求:根据 menu_level,sort排序,越小的越前面. -- 下面代码按照升序规则进行 ...

  7. java8新特性:三,Stream

    java8新特性:三,Stream 1 Stream介绍 Java 8 API添加了一个新的抽象称为流Stream,可以让你以一种声明的方式处理数据. Stream 使用一种类似用 SQL 语句从数据 ...

  8. java8新特性(4)— Stream流

    java8新特性(4)- Stream流 遍历集合更强大 package com.common.jdk8;import java.util.*; import java.util.stream.Col ...

  9. java8 lambda 接口_Java8新特性之一:Lambda表达式

    Java8是自java5之后最重大的一次更新,它给JAVA语言带来了很多新的特性(包括编译器.类库.工具类.JVM等),其中最重要的升级是它给我们带来了Lambda表达式和Stream API. 1. ...

最新文章

  1. 如何将cocos2d-x程序分别移植到ios,android,windowsphone三个手机平台上
  2. corosync+pacemaker+crm简单配置
  3. SIC插槽,WSIC插槽,XSIC插槽
  4. oracle12c多个pdb,Oracle 12c 多租户专题|12cR2中PDB内存资源管理
  5. Visual Studio 2017 社区版的安装与组件修改(C++)
  6. 【图论】【斜率优化】前往大都会(loj 2769)
  7. c# 自定义应用程序配置文件(app.config)
  8. 语音识别插件_AnsweringMachine XS: 越狱理由之二,iPhone 电话语音答录机
  9. 在计算机系统中 外存储器必须通过,大学计算机基础第4章作业.doc
  10. 剑指offer——面试题33:把数组排成最小数
  11. 如何修改默认字体_Excel技巧:怎么修改默认字体为宋体
  12. 计算机网络(谢希仁)第六版课件
  13. 1024程序节|Android框架之一 BRVAH【BaseRecyclerViewAdapterHelper】使用demo
  14. pip install 安装requirements.txt经常报错解决方法
  15. 华硕主板装系统蓝屏_装xp系统蓝屏,电脑安装xp系统蓝屏怎么办
  16. 基于新浪微博的男女性择偶观数据分析(下)
  17. [递推] 费解的开关
  18. 中盈Zonewin NX-1900 打印机驱动
  19. 利用Python绘制小狗小猫
  20. 【安全算法】一文带你简要了解常见常用的安全算法

热门文章

  1. linux下tomcat脚本,Linux下重启多个 tomcat 服务的脚本(推荐)
  2. mysql设置输出格式_rsyslog 配置mysql输出格式
  3. linux安装python2环境_Python基础手册 2 —— Python 环境搭建(Linux)
  4. apex英雄机器人探路者怎么玩_Apex英雄探路者机器人实战技巧攻略[多图]
  5. 开机按f12怎么恢复系统 开机按f12恢复系统
  6. Git Permission to fazhiyun86/Test.git denied to MarRoar
  7. Activiti 基础概念
  8. SpringAop与AspectJ的联系与区别____比较分析 Spring AOP 和 AspectJ 之间的差别
  9. Redis之Redis的事务
  10. Java面试——RabbitMQ系列总结