一直在写中间件相关的代码,提供SDK给业务方使用,但很多业务方还一直停留在1.7版本,迟迟不升级,为了兼容性,不敢在代码中使用Java8的一些新特性,比如Stream之类的,虽然不能用,但还是要学一下。

Stream 是什么

Stream 是Java 8中添加的一个新特性,它与 java.io 包里的 InputStream 和 OutputStream 是完全不同的概念。它借助于 Lambda 表达式,可以让你以一种声明的方式处理数据,可以极大提高Java程序员的生产力,让程序员写出高效率、干净、简洁的代码。

Stream Demo

直接上Demo,感受一下

  1. List<String> myList = Arrays.asList("a", "b", "c", "d", "e");

  2. myList.stream()

  3. .filter(s -> s.startsWith("1"))

  4. .map(String::toUpperCase)

  5. .sorted()

  6. .forEach(System.out::println);

Stream 如何工作

当使用一个流的时候,通常包括三个基本步骤:

  • 获取一个数据源(source)

  • 数据转换

  • 执行操作获取想要的结果

每次转换原有 Stream 对象不改变,返回一个新的 Stream 对象(可以有多次转换),这就允许对其操作可以像链条一样排列,变成一个管道,如下图所示。

在Stream中,分为两种操作

  • 中间操作

  • 结束操作

中间操作返回Stream,终端操作返回void或者非Stream结果,在demo中, filter、 map、 sorted都算是中间操作,而 forEach是一个结束操作。

Stream 如何生成

创建Stream的方式很多,最常见的是从Collections,List 和 Set中生成

  1. List<String> myList = Arrays.asList("a1", "a2", "b1", "c2", "c1");

  2. Stream<String> stream = myList.stream()

在对象myList上调用方法 stream() 返回一个常规对象Stream。

也可以从一堆已知对象中生成。

  1. Stream<String> stream = Stream.of("a1", "a2", "a3")

当然了,还有其它方式:

  • Collection.stream()

  • Collection.parallelStream()

  • BufferedReader.lines()

  • Files.walk()

  • BitSet.stream()

  • Random.ints()

  • JarFile.stream()

  • ....

常规操作

forEach

forEach方法接收一个 Lambda 表达式,用来迭代流中的每个数据

  1. Stream.of(1, 2, 3).forEach(System.out::println);

  2. // 1

  3. // 2

  4. // 3

map

map 用于映射每个元素到对应的结果

  1. Stream.of(1, 2, 3).map( i -> i*i).forEach(System.out::println);

  2. // 1

  3. // 4

  4. // 9

filter

filter 用于通过设置的条件过滤出元素

  1. Stream.of(1, 2, 3).filter( i -> i == 1).forEach(System.out::println);

  2. // 1

limit

limit 用于用于获取指定数量的流

  1. Stream.of(1, 2, 3, 4, 5).limit(2).forEach(System.out::println);

  2. // 1

  3. // 2

sorted

sorted 用于对流进行排序

  1. Stream.of(4, 1, 5).sorted().forEach(System.out::println);

  2. // 1

  3. // 4

  4. // 5

Match

有三个 match 方法,从语义上说:

  • allMatch:Stream 中全部元素符合传入的 predicate,返回 true

  • anyMatch:Stream 中只要有一个元素符合传入的 predicate,返回 true

  • noneMatch:Stream 中没有一个元素符合传入的 predicate,返回 true

它们都不是要遍历全部元素才能返回结果。例如 allMatch 只要一个元素不满足条件,就 skip 剩下的所有元素,返回 false。

  1. boolean result = Stream.of("a1", "a2", "a3").allMatch(i -> i.startsWith("a"));

  2. System.out.println(result);

  3. // true

reduce

reduce 方法根据指定的函数将元素序列累积到某个值。此方法有两个参数:

  • 起始值

  • 累加器函数。

如果有一个List,希望得到所有这些元素和一些初始值的总和。

  1. int result = Stream.of(1, 2, 3).reduce(20, (a,b) -> a + b);

  2. System.out.println(result);

  3. // 26

collect

Collectors类中提供了功能丰富的工具方法

  • toList

  • toSet

  • toCollection

  • toMap

  • ...

而这些方法,都需要通过 collect 方法传入。

  1. Set<Integer> result = Stream.of(1, 1, 2, 3).collect(Collectors.toSet());

  2. System.out.println(result);

  3. // [1, 2, 3]

collect 可以把Stream数据流转化为Collection对象,

骚技巧

for循环

除了常规的对象Stream,还有一些有特殊类型的Stream,用于处理基本数据类型int、long和double,它是IntStream、LongStream和DoubleStream。

比如可以使用IntStream.range()来代替常规的for循环。

  1. IntStream.range(1, 4).forEach(System.out::println);

随机数

Random的ints方法可以返回一个随机数据流,比如返回1到100的10个随机数。

  1. Random random = new Random();

  2. random.ints(1, 100).limit(10).forEach(System.out::println);

大小写转化

  1. List<String> output = wordList.stream()

  2. .map(String::toUpperCase)

  3. .collect(Collectors.toList());

Stream 特点

总之,Stream 的特性可以归纳为:

无存储

Stream并不是一种数据结构,它只是某种数据源的一个视图

安全性

对Stream的任何修改都不会修改背后的数据源,比如对stream执行过滤操作并不会删除被过滤的元素,而是会产生一个不包含被过滤元素的新Stream。

惰式执行

Stream上的操作并不会立即执行,只有等到用户真正需要结果的时候才会执行。

一次性

Stream只能被“消费”一次,一旦遍历过就会失效,就像容器的迭代器那样,想要再次遍历必须重新生成。

lambda

所有 Stream 的操作必须以 lambda 表达式为参数

转 java 8 lamba stream的更多相关文章

  1. 详解Java 8中Stream类型的“懒”加载

    在进入正题之前,我们需要先引入Java 8中Stream类型的两个很重要的操作: 中间和终结操作(Intermediate and Terminal Operation) Stream类型有两种类型的 ...

  2. java.io.IOException: Stream closed

    今天在做SSH项目的时候,出现了这个错误.百思不得其解,网上的答案都不能解决我的问题-.. 后来,一气之下就重新写,写了之后发现在JSP遍历集合的时候出错了. <s:iterator value ...

  3. Java基础(十一) Stream I/O and Files

    Java基础(十一) Stream I/O and Files 1. 流的概念 程序的主要任务是操纵数据.在Java中,把一组有序的数据序列称为流. 依据操作的方向,能够把流分为输入流和输出流两种.程 ...

  4. java.io.IOException: Stream closed解决办法

    1.出现这个bug的大体逻辑代码如下: private static void findMovieId() throws Exception { File resultFile = new File( ...

  5. Java 8创建Stream流的5种方法

    不知不觉间,Java已经发展到13了,来不及感慨时间过得真的太快了,来不及学习日新月异的技术更新,目前大多数公司还是使用的JDK8版本,一方面是版本的稳定,另一方面是熟悉,所以很多公司都觉得不升级也挺 ...

  6. java中的Stream流

    java中的Stream流 说到Stream便容易想到I/O Stream,而实际上,谁规定"流"就一定是"IO流"呢?在Java 8中,得益于Lambda所带 ...

  7. java新特性stream

    java新特性stream,也称为流式编程. 在学习stream之前先了解一下java内置的四大函数 第一种函数式函数,后面是lambda表达式写法 /*Function<String,Inte ...

  8. Java基础系列-Stream

    原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/10748817.html 一.概述 Stream操作简称流操作,这里的流与IO流毫无关系, ...

  9. Java 8中Stream API学习笔记

    1)函数式编程的优势和劣势分别是什么?优势:①不可变性 ②并行操作 ③执行顺序更灵活 ④代码更加简洁纯粹的函数式编程,变量具有不可变性,同一个参数不会在不同场景下得出不同的结果,因此大大增强了系统的稳 ...

随机推荐

  1. Photon介绍与安装

    Photon光子引擎是一款实时的Socket服务器和开发框架,快速.使用方便.容易扩展. 服务端架构在windows系统平台上,采用C#语言编写. 客户端SDK提供了多种平台的开发API,包括DotN ...

  2. sparkjob的提交流程

    在使用spark-submit提交一个Spark应用之后,Driver程序会向集群申请一定的资源来启动东若干个Executors用来计算,当这些Executors启动后,它们会向Driver端的Sch ...

  3. React Tutorial: Basic Concept Of React Component---babel, a translator

    Getting started with react.js: basic concept of React component 1 What is React.js React, or React.j ...

  4. Quick Start NodeMCU / ESP8266 12E

    先说明一下:本来想买常见的ESP 8266作为Arduinoi的WIFI模块,结果错买成ESP 8266 12E,发现网上的资料比较少. ESP8266是WIFI芯片,它只是一块芯片必须要搭配相应的电 ...

  5. 11.2Test

    11.2Test 题目 描述 做法 \(BSOJ6367\) 求一个字符串内出现最多长度为\(K\)的子串 定长\(Hash\) \(BSOJ6368\) 给出数列\(a_{i+2}=ka_{i+1} ...

  6. Numpy | 09 高级索引

    NumPy 比一般的 Python 序列提供更多的索引方式.除了之前看到的用整数和切片的索引外,数组可以由整数数组索引.布尔索引及花式索引. 整数数组索引 实例1:获取数组中(0,0),(1,1)和( ...

  7. mark一下咕掉的题目

    蒟蒻才普及组呀~ 大佬别D我 等集中补一下 CF980F:咋说捏,我觉得要联赛了做这题有点浪费时间,等想颓废了再来写写叭qwq 215E/279D/288E/331C3/431D/433E/750G/ ...

  8. javaweb利用filter拦截未授权请求

    项目上有个小需求,要限制访问者的IP,屏蔽未授权的请求.该场景使用过滤器来做再合适不过了. SecurityFilter.java: public class SecurityFilter imple ...

  9. hdu2037 今年暑假不AC[贪心][区间调度问题]

    目录 题目地址 题干 代码和解释 参考 题目地址 hdu2037 题干 代码和解释 本题使用贪心.有三种贪心策略:开始时间最早,结束时间最早,用时最短.第二种是正确的策略,因为结束得越早,后面就可以有 ...

  10. Pandas 与 Numpy 常用方法总结

    Lambda 函数实现 简单的说,lambda 就是一个函数,但是这个函数没有名字,所以我们介绍一下这个函数的调用形式,参数与返回值的实现. lambda 的格式如下: lambda [arg1 [, ...