从 01 开始 从 01 开始
首页
  • 📚 计算机基础

    • 计算机简史
    • 数字电路
    • 计算机组成原理
    • 操作系统
    • Linux
    • 计算机网络
    • 数据库
    • 编程工具
    • 装机
  • 🎨 前端

    • Node
  • JavaSE
  • Java 高级
  • JavaEE

    • 构建、依赖管理
    • Ant
    • Maven
    • 日志框架
    • Junit
    • JDBC
    • XML-JSON
  • JavaWeb

    • 服务器软件
    • 环境管理和配置管理-科普篇
    • Servlet
  • Spring

    • Spring基础
  • 主流框架

    • Redis
    • Mybatis
    • Lucene
    • Elasticsearch
    • RabbitMQ
    • MyCat
    • Lombok
  • SpringMVC

    • SpringMVC 基础
  • SpringBoot

    • SpringBoot 基础
  • Windows 使用技巧
  • 手机相关技巧
  • 最全面的输入法教程
  • 最全面的浏览器教程
  • Office
  • 图片类工具
  • 效率类工具
  • 最全面的 RSS 教程
  • 码字工具
  • 各大平台
  • 校招
  • 五险一金
  • 职场规划
  • 关于离职
  • 杂谈
  • 自媒体
  • 📖 读书

    • 读书工具
    • 走进科学
  • 🌍 英语

    • 从零开始学英语
    • 英语兔的相关视频
    • Larry 想做技术大佬的相关视频
  • 🏛️ 政治

    • 新闻合订本
    • 反腐
    • GFW
    • 404 内容
    • 审查与自我审查
    • 互联网
    • 战争
    • 读书笔记
  • 💰 经济

    • 关于税
    • 理财
  • 💪 健身

    • 睡眠
    • 皮肤
    • 口腔健康
    • 学会呼吸
    • 健身日志
  • 🏠 其他

    • 驾驶技能
    • 租房与买房
    • 厨艺
  • 电影

    • 电影推荐
  • 电视剧
  • 漫画

    • 漫画软件
    • 漫画推荐
  • 游戏

    • Steam
    • 三国杀
    • 求生之路
  • 小说
  • 关于本站
  • 关于博主
  • 打赏
  • 网站动态
  • 友人帐
  • 从零开始搭建博客
  • 搭建邮件服务器
  • 本站分享
  • 🌈 生活

    • 2022
    • 2023
    • 2024
    • 2025
  • 📇 文章索引

    • 文章分类
    • 文章归档

晓林

程序猿,自由职业者,博主,英语爱好者,健身达人
首页
  • 📚 计算机基础

    • 计算机简史
    • 数字电路
    • 计算机组成原理
    • 操作系统
    • Linux
    • 计算机网络
    • 数据库
    • 编程工具
    • 装机
  • 🎨 前端

    • Node
  • JavaSE
  • Java 高级
  • JavaEE

    • 构建、依赖管理
    • Ant
    • Maven
    • 日志框架
    • Junit
    • JDBC
    • XML-JSON
  • JavaWeb

    • 服务器软件
    • 环境管理和配置管理-科普篇
    • Servlet
  • Spring

    • Spring基础
  • 主流框架

    • Redis
    • Mybatis
    • Lucene
    • Elasticsearch
    • RabbitMQ
    • MyCat
    • Lombok
  • SpringMVC

    • SpringMVC 基础
  • SpringBoot

    • SpringBoot 基础
  • Windows 使用技巧
  • 手机相关技巧
  • 最全面的输入法教程
  • 最全面的浏览器教程
  • Office
  • 图片类工具
  • 效率类工具
  • 最全面的 RSS 教程
  • 码字工具
  • 各大平台
  • 校招
  • 五险一金
  • 职场规划
  • 关于离职
  • 杂谈
  • 自媒体
  • 📖 读书

    • 读书工具
    • 走进科学
  • 🌍 英语

    • 从零开始学英语
    • 英语兔的相关视频
    • Larry 想做技术大佬的相关视频
  • 🏛️ 政治

    • 新闻合订本
    • 反腐
    • GFW
    • 404 内容
    • 审查与自我审查
    • 互联网
    • 战争
    • 读书笔记
  • 💰 经济

    • 关于税
    • 理财
  • 💪 健身

    • 睡眠
    • 皮肤
    • 口腔健康
    • 学会呼吸
    • 健身日志
  • 🏠 其他

    • 驾驶技能
    • 租房与买房
    • 厨艺
  • 电影

    • 电影推荐
  • 电视剧
  • 漫画

    • 漫画软件
    • 漫画推荐
  • 游戏

    • Steam
    • 三国杀
    • 求生之路
  • 小说
  • 关于本站
  • 关于博主
  • 打赏
  • 网站动态
  • 友人帐
  • 从零开始搭建博客
  • 搭建邮件服务器
  • 本站分享
  • 🌈 生活

    • 2022
    • 2023
    • 2024
    • 2025
  • 📇 文章索引

    • 文章分类
    • 文章归档
  • JavaSE

    • 我的 Java 学习路线
    • 安装 Java
    • Java 数据类型

    • Java 多版本配置
    • 面向对象

    • Java 核心类

    • IO

    • Java 与时间

    • 异常处理

    • 哈希和加密算法

    • Java8 新特性

      • 函数式编程与 Lambda
      • Stream 介绍
        • 什么是 Stream
        • 在编程中使用 Stream
        • Stream 的本质:计算规则
        • 小结
      • 创建 Stream
      • 操作 Stream
      • Optional
    • 网络编程

  • JavaSenior

  • JavaEE

  • JavaWeb

  • Spring

  • 主流框架

  • SpringMVC

  • SpringBoot

  • Java
  • JavaSE
  • Java8 新特性
2023-03-13
目录

Stream 介绍

# Stream 介绍

Java 从 8 开始,不但引入了 Lambda 表达式,还引入了一个全新的流式 API:Stream API。它位于 java.util.stream 包中。

声明:本文主要参考廖雪峰老师的教程 使用 Stream (opens new window),并自己动手实践了一遍。

# 什么是 Stream

重点:这个 Stream 不同于 java.io 的 InputStream 和 OutputStream,它代表的是任意 Java 对象的序列。两者对比如下:

java.io java.util.stream
存储 顺序读写的 byte 或 char 顺序输出的任意 Java 对象实例
用途 序列化至文件或网络 内存计算/业务逻辑

有同学会问:一个顺序输出的 Java 对象序列,不就是一个 List 容器吗?

再次划重点:这个 Stream 和 List 也不一样,List 存储的每个元素都是已经存储在内存中的某个 Java 对象,而 Stream 输出的元素可能并没有预先存储在内存中,而是实时计算出来的。

换句话说,List 的用途是操作一组已存在的 Java 对象,而 Stream 实现的是惰性计算,用到的时候再计算。两者对比如下:

java.util.List java.util.stream
元素 已分配并存储在内存 可能未分配,实时计算
用途 操作一组已存在的 Java 对象 惰性计算

Stream 看上去有点不好理解,但我们举个例子就明白了

List 就好比自助餐,东西都已经做好了,想吃什么就直接去拿就可以

stream 就好比普通的饭店,先点菜后再做饭,做好了才可以吃

# 在编程中使用 Stream

我们来动手实践下,例如我们要表示一个全体自然数的集合,显然,用 List 是不可能写出来的,因为自然数是无限的,内存再大也没法放到 List 中。

但是,用 Stream 可以做到。写法如下:

Stream<BigInteger> naturals = createNaturalStream(); // 全体自然数
1

我们先不考虑 createNaturalStream() 这个方法是如何实现的,我们看看如何使用这个 Stream。

首先,我们可以对每个自然数做一个平方,这样我们就把这个 Stream转换成了另一个 Stream:

Stream<BigInteger> naturals = createNaturalStream(); // 全体自然数
Stream<BigInteger> streamNxN = naturals.map(n -> n.multiply(n)); // 全体自然数的平方
1
2

因为这个 streamNxN 也有无限多个元素,要打印它,必须首先把无限多个元素变成有限个元素,可以用 limit() 方法截取前 100 个元素,最后用 forEach() 处理每个元素,这样,我们就打印出了前 100 个自然数的平方:

Stream<BigInteger> naturals = createNaturalStream();
naturals.map(n -> n.multiply(n)) // 1, 4, 9, 16, 25...
        .limit(100)
        .forEach(System.out::println);
1
2
3
4

我们总结一下 Stream的特点:它可以“存储”有限个或无限个元素。这里的存储打了个引号,是因为元素有可能已经全部存储在内存中,也有可能是根据需要实时计算出来的。

# Stream 的本质:计算规则

Stream 可以理解为只存储了计算规则,需要输出元素的时候,才进行计算。

Stream 的另一个特点是,一个 Stream 可以轻易地转换为另一个 Stream,而不是修改原 Stream 本身。

最后,真正的计算通常发生在最后结果的获取,也就是惰性计算。

Stream<BigInteger> naturals = createNaturalStream(); // 不计算
Stream<BigInteger> s2 = naturals.map(BigInteger::multiply); // 不计算
Stream<BigInteger> s3 = s2.limit(100); // 不计算
s3.forEach(System.out::println); // 计算
1
2
3
4

惰性计算的特点是:一个 Stream 转换为另一个 Stream 时,实际上只存储了计算规则,并没有任何计算发生。

例如,创建一个全体自然数的 Stream,不会进行计算,把它转换为上述 s2 这个 Stream,也不会进行计算。再把 s2 这个无限 Stream 转换为 s3 这个有限的 Stream,也不会进行计算。只有最后,调用 forEach 确实需要 Stream 输出的元素时,才进行计算。我们通常把 Stream 的操作写成链式操作,代码更简洁:

createNaturalStream()
    .map(BigInteger::multiply)
    .limit(100)
    .forEach(System.out::println);
1
2
3
4

因此,Stream API 的基本用法就是:创建一个 Stream,然后做若干次转换,最后调用一个求值方法获取真正计算的结果:

int result = createNaturalStream() // 创建Stream
             .filter(n -> n % 2 == 0) // 任意个转换
             .map(n -> n * n) // 任意个转换
             .limit(100) // 任意个转换
             .sum(); // 最终计算结果
1
2
3
4
5

# 小结

Stream API 的特点是:

  • Stream API 提供了一套新的流式处理的抽象序列;
  • Stream API 支持函数式编程和链式操作;
  • Stream 可以表示无限序列,并且大多数情况下是惰性求值的。
上次更新: 2025/6/3 09:31:54
函数式编程与 Lambda
创建 Stream

← 函数式编程与 Lambda 创建 Stream→

最近更新
01
新闻合订本 2025-10
10-31
02
2025 年 10 月记
10-30
03
用 AI 批量优化思源笔记排版
10-15
更多文章>
Theme by Vdoing | Copyright © 2022-2025 | 粤 ICP 备 2022067627 号 -1 | 粤公网安备 44011302003646 号 | 点击查看十年之约
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式