影子

在学习Spring WebFlux之前,我们先来了解JDK的Stream,虽然他们之间没有直接的关系,有趣的是 Spring Web Flux 基于 Reactive Stream,他们中都带了 Stream。现有需求如下:筛选出一个数组中的偶数,每个增加 100 后输出到控制台,我们来看下使用JDK Stream和使用Reactor(Reactive Stream的一种实现,后面会讲)编写的代码:

  1. // JDK Stream实现
  2. Arrays.stream(ARRAY)
  3. .filter(num -> num % 2 == 0)
  4. .map(num -> num + 100)
  5. .forEach(System.out::println);
  1. // Reactor实现
  2. Flux.fromArray(ARRAY)
  3. .filter(num -> num % 2 == 0)
  4. .map(num -> num + 100)
  5. .subscribe(System.out::println);

以上代码见JdkStreamAndReactorTest。

我们发现他们的写法是相似的,都是采用函数式编程,并且其中有很多函数(操作符)是一样的,抛除他们的异同点(后面会讲),我们先来了解下 Java8 的 Stream API,方便我们后面可以更快的了解 Reactor 中的各种操作符。

Stream API

Java8中有两大最为重要的改变:第一个是 Lambda 表达式;另外一个则是 Stream API(java.util.stream.*)。

Stream Api 位于 java.util.stream 包下,Stream接口是该包下的关键抽象。Stream、IntStream、LongStream、DoubleStream 分别是Object(支持泛型))、int(整数)、long(长整数)、double(双精度浮点数)类型的数据流抽象。

流在如下几个方面与集合不同:

  • 不做存储。流不是存储元素的数据结构。相反,它从诸如数据结构、数组、生成函数或I/O通道之类的源中通过一系列计算操作(向流管道中)传递元素。
  • 函数式的。对流的操作会产生结果,但不会修改源数据。例如,对从集合中获取的Stream进行过滤会生成一个不带过滤元素的新Stream,而不是从源集合中删除元素。
  • 懒惰操作。许多流操作(例如过滤,映射或重复删除)总是延迟操作,从而找到优化的机会。例如,“查找以某三个字母开头的第一个字符串”不需要检查所有输入字符串。流操作分为中间(流产生)操作和终端(产生值或副作用)操作。中间操作总是惰性的。
  • 可能无界。尽管集合的大小是有限的,但流不是必需的。诸如 limit(n) 或 findFirst() 之类的短路操作可以允许对无限流的计算在有限时间内完成。
  • 消耗品。在流的生存期内,流的元素只能访问一次。像Iterator一样,必须生成新的流以重新访问源中的相同元素。

Stream 是 Java8 中处理集合的关键抽象概念,它可以指定你希望对集合进行的操作,可以执行非常复杂的查找、过滤和映射数据等操作。使用Stream API 对集合数据进行操作,就类似于使用 SQL 执行的数据库查询。也可以使用 Stream API 来并行执行操作。简而言之,Stream API 提供了一种高效且易于使用的处理数据的方式。

Stream的用法

1. 创建 Stream

一个数据源(如:集合、数组), 获取一个流。

2. 中间操作

一个中间操作链,对数据源的数据进行处理。

3. 终端操作(终止操作)

一个终止操作,执行中间操作链,并产生结果 。

Stream的分类

序号 类名 说明
1 BaseStream Stream接口的父接口
2 Stream 泛型类型的Stream
3 IntStream 整形Stream
4 LongStream 长整型Stream
5 DoubleStream 浮点型Stream

序号3-5的具体类型Stream提供了一些额外的方法,在下面章节【创建Stream】时有用到。它们之间的类图关系如下:

创建Stream

创建Stream主要分为如下几种方式:

  1. 集合获取Stream
  2. 数组创建Stream
  3. 值创建Stream
  4. 函数创建Stream
  5. 其他方式创建Stream

由于篇幅问题及考虑到Stream提供的操作方法没有讲解,下面的示例代码中只是提供了创建Stream的示例,具体的使用示例请看代码。

注意:以下创建Stream的方式仅为演示使用,因为Stream进行中间操作或终端操作后就会关闭,不可重复使用,因此你在使用的时候应该按函数式编程方式编写代码。

1. 集合获取Stream

  1. // 返回以此集合作为源的顺序 Stream
  2. Stream<Integer> stream = collection.stream();
  3. // 返回以此集合作为其来源可能并行的 Stream
  4. Stream<Integer> parallelStream = collection.parallelStream();

2. 数组创建Stream

  1. // 创建数组Stream
  2. Stream<String> stream = Arrays.stream(strArray);
  3. IntStream intStream = Arrays.stream(intArray);
  4. // 数组Stream的重载
  5. DoubleStream doubleStream = Arrays.stream(doubleArray);
  6. IntStream intStream2 = Arrays.stream(intArray, 1, 3);

3. 值创建Stream

  1. // 构建Integer类型的Stream
  2. IntStream stream = IntStream.of(14, 2, 31, 47, 5, 6, 9, 1, 33, 2, 6);
  3. // 构建String类型的Stream
  4. Stream<String> stringStream = Stream.of("Hello, Stream Api.");

4. 函数创建Stream

  1. // 方式1:使用generate方式创建一个新的无限无序Stream流
  2. Stream<Integer> generateStream = Stream.generate(RandomUtil::randomInt);
  3. // 方式2:使用iterate方式创建一个新的无限有序Stream流
  4. IntStream iterateStream = IntStream.iterate(1, n -> n + 1);

5. 其他方式创建Stream

  1. // 方式1:创建空的顺序流
  2. Stream<Object> emptyStream = Stream.empty();
  3. // 方式2:使用两个Stream创建组合Stream
  4. IntStream concatStream = IntStream.concat(intStream1, intStream2);
  5. // 方式3:创建begin至end逐渐加1的整形Stream
  6. IntStream rangeStream = IntStream.range(1, 501);
  7. // 方式4:使用建造者模式创建Stream
  8. Stream.Builder<Integer> builder = Stream.builder();
  9. Stream<Integer> buildStream = builder.build();
  10. // 方式5:使用StreamSupport创建Stream
  11. // 代码忽略,具体使用请看API...

以上代码见CreateStreamTest。

源码详见:https://github.com/crystalxmumu/spring-web-flux-study-note

以上是本次笔记的内容,我们下次见。

参考

  1. 【Java8新特性】关于Java8的Stream API,看这一篇就够了

Spring WebFlux 学习笔记 - (一) 前传:学习Java 8 Stream Api (1) - 创建 Stream的更多相关文章

  1. Spring WebFlux 学习笔记 - (一) 前传:学习Java 8 Stream Api (2) - Stream的中间操作

    Stream API Java8中有两大最为重要的改变:第一个是 Lambda 表达式:另外一个则是 Stream API(java.util.stream.*). Stream 是 Java8 中处 ...

  2. Spring WebFlux 学习笔记 - (一) 前传:学习Java 8 Stream Api (3) - Stream的终端操作

    Stream API Java8中有两大最为重要的改变:第一个是 Lambda 表达式:另外一个则是 Stream API(java.util.stream.*). Stream 是 Java8 中处 ...

  3. 【SpringSecurity系列3】基于Spring Webflux集成SpringSecurity实现前后端分离无状态Rest API的权限控制

    源码传送门: https://github.com/ningzuoxin/zxning-springsecurity-demos/tree/master/02-springsecurity-state ...

  4. Vue学习笔记-Vue.js-2.X 学习(一)===>基本知识学习

    一  使用环境: windows 7 64位操作系统 二  IDE:VSCode/PyCharm 三  Vue.js官网: https://cn.vuejs.org/ 四  下载安装引用 方式1:直接 ...

  5. [原创]java WEB学习笔记75:Struts2 学习之路-- 总结 和 目录

    本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...

  6. [原创]java WEB学习笔记66:Struts2 学习之路--Struts的CRUD操作( 查看 / 删除/ 添加) 使用 paramsPrepareParamsStack 重构代码 ,PrepareInterceptor拦截器,paramsPrepareParamsStack 拦截器栈

    本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...

  7. 【神经网络与深度学习】学习笔记:AlexNet&Imagenet学习笔记

    学习笔记:AlexNet&Imagenet学习笔记 ImageNet(http://www.image-net.org)是李菲菲组的图像库,和WordNet 可以结合使用 (毕业于Caltec ...

  8. Vue学习笔记-Vue.js-2.X 学习(三)===>组件化高级

    (四) 组件化高级 1.插槽(slot)的基本使用 A:基本使用: <slot></slot> B:默认置:<slot><h1>中间可以放默认值< ...

  9. Vue学习笔记-Vue.js-2.X 学习(二)===>组件化开发

    ===重点重点开始 ========================== (三) 组件化开发 1.创建组件构造器: Vue.extends() 2.注册组件: Vue.component() 3.使用 ...

随机推荐

  1. 使用vue实现购物车功能

    页面效果图: html代码: <div class="shop-car" id='car'> <div class="count-custom" ...

  2. MySQL银行ATM存取款机系统(需求分析)

    银行ATM需求文档 一.E-R图形文 二.开发步骤 1.明确需求--数据库设计--编码实现功能--测试 2.绘制e-r图--绘制数据库模型图--使用三大方式规范数据库结构 三.开发思路 1. 模型图综 ...

  3. python3.x 基础七:面向对象进阶

    类的高级方法: 1.静态方法:在类方法前增加关键字@staticmethod,将普通方法变成静态方法,不能再次传值,不能访问实例变量或者类变量,与类的关系仅仅是通过类名进行调用 2.类方法:在类方法前 ...

  4. Java入门以及使用IDEA安装和使用

    目录 JAVA第一个程序 HelloWorld 可能会遇到的情况 JAVA程序运行机制 IDEA安装 什么是IDE IDEA介绍 IDEA下载 IDEA基本使用 关于一些创建项目的小tips JAVA ...

  5. Cypress系列(1)- Window下安装 Cypress 并打开

    如果想从头学起Cypress,可以看下面的系列文章哦 https://www.cnblogs.com/poloyy/category/1768839.html 系统要求 Cypress 是一个被安装在 ...

  6. Qt版本中国象棋开发(四)

    内容:走法产生 中国象棋基础搜索AI, 极大值,极小值剪枝搜索, 静态估值函数 理论基础: (一)人机博弈走法产生: 先遍历某一方的所有棋子,再遍历整个棋盘,得到每个棋子的所有走棋情况(效率不高,可以 ...

  7. 【深度学习】PyTorch之Squeeze()和Unsqueeze()

    1. unsqueeze() 该函数用来增加某个维度.在PyTorch中维度是从0开始的. import torch a = torch.arange(0, 9) print(a) 结果: tenso ...

  8. 实用教程丨使用K3s和MySQL运行Rancher 2.4

    本文转自Rancher Labs 简 介 本文将介绍在高可用K3s Kubernetes集群上安装Rancher 2.4的过程并针对MySQL利用Microsoft Azure数据库的优势,该数据库消 ...

  9. vc程序设计-----位图

    // resource_study.cpp : 定义应用程序的入口点. // #include "framework.h" #include "resource_stud ...

  10. SRAM电路工作原理

    近年来,片上存储器发展迅速,根据国际半导体技术路线图(ITRS),随着超深亚微米制造工艺的成熟和纳米工艺的发展,晶体管特征尺寸进一步缩小,半导体存储器在片上存储器上所占的面积比例也越来越高.接下来宇芯 ...