写在前面

先说点题外话:不少读者工作几年后,仍然在使用Java7之前版本的方法,对于Java8版本的新特性,甚至是Java7的新特性几乎没有接触过。真心想对这些读者说:你真的需要了解下Java8甚至以后版本的新特性了。

今天,一名读者出去面试,面试官问他:说说Java8中创建Stream流有哪几种方式?他竟然没回答上来!!

Stream概述

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

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

何为Stream?

流(Stream) 到底是什么呢?

可以这么理解流:流就是数据渠道,用于操作数据源(集合、数组等)所生成的元素序列。

“集合讲的是数据,流讲的是计算!”

注意:

①Stream 自己不会存储元素。

②Stream 不会改变源对象。相反,他们会返回一个持有结果的新Stream。

③Stream 操作是延迟执行的。这意味着他们会等到需要结果的时候才执行。

Stream操作步骤

1.创建 Stream

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

2.中间操作

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

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

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

如何创建Stream流?

这里,创建测试类TestStreamAPI1,所有的操作都是在TestStreamAPI1类中完成的。

(1)通过Collection系列集合提供的stream()方法或者parallelStream()方法来创建Stream。

在Java8中,Collection 接口被扩展,提供了两个获取流的默认方法,如下所示。

default Stream<E> stream() {return StreamSupport.stream(spliterator(), false);
}
default Stream<E> parallelStream() {return StreamSupport.stream(spliterator(), true);
}

其中,stream()方法返回一个顺序流,parallelStream()方法返回一个并行流。

我们可以使用如下代码方式来创建顺序流和并行流。

List<String> list = new ArrayList<>();
list.stream();
list.parallelStream();

(2)通过Arrays中的静态方法stream()获取数组流。

Java8 中的 Arrays类的静态方法 stream() 可以获取数组流 ,如下所示。

public static <T> Stream<T> stream(T[] array) {return stream(array, 0, array.length);
}

上述代码的的作用为:传入一个泛型数组,返回这个泛型的Stream流。

除此之外,在Arrays类中还提供了stream()方法的如下重载形式。

public static <T> Stream<T> stream(T[] array) {return stream(array, 0, array.length);
}public static <T> Stream<T> stream(T[] array, int startInclusive, int endExclusive) {return StreamSupport.stream(spliterator(array, startInclusive, endExclusive), false);
}public static IntStream stream(int[] array) {return stream(array, 0, array.length);
}public static IntStream stream(int[] array, int startInclusive, int endExclusive) {return StreamSupport.intStream(spliterator(array, startInclusive, endExclusive), false);
}public static LongStream stream(long[] array) {return stream(array, 0, array.length);
}public static LongStream stream(long[] array, int startInclusive, int endExclusive) {return StreamSupport.longStream(spliterator(array, startInclusive, endExclusive), false);
}public static DoubleStream stream(double[] array) {return stream(array, 0, array.length);
}public static DoubleStream stream(double[] array, int startInclusive, int endExclusive) {return StreamSupport.doubleStream(spliterator(array, startInclusive, endExclusive), false);
}

基本上能够满足基本将基本类型的数组转化为Stream流的操作。

我们可以通过下面的代码示例来使用Arrays类的stream()方法来创建Stream流。

Integer[] nums = new Integer[]{1,2,3,4,5,6,7,8,9};
Stream<Integer> numStream = Arrays.stream(nums);

(3)通过Stream类的静态方法of()获取数组流。

可以使用静态方法 Stream.of(), 通过显示值创建一个流。它可以接收任意数量的参数。

我们先来看看Stream的of()方法,如下所示。

public static<T> Stream<T> of(T t) {return StreamSupport.stream(new Streams.StreamBuilderImpl<>(t), false);
}
@SafeVarargs
@SuppressWarnings("varargs")
public static<T> Stream<T> of(T... values) {return Arrays.stream(values);
}

可以看到,在Stream类中,提供了两个of()方法,一个只需要传入一个泛型参数,一个需要传入一个可变泛型参数。

我们可以使用下面的代码示例来使用of方法创建一个Stream流。

Stream<String> strStream = Stream.of("a", "b", "c");

(4)创建无限流

可以使用静态方法 Stream.iterate() 和Stream.generate(), 创建无限流。

先来看看Stream类中iterate()方法和generate()方法的源码,如下所示。

public static<T> Stream<T> iterate(final T seed, final UnaryOperator<T> f) {Objects.requireNonNull(f);final Iterator<T> iterator = new Iterator<T>() {@SuppressWarnings("unchecked")T t = (T) Streams.NONE;@Overridepublic boolean hasNext() {return true;}@Overridepublic T next() {return t = (t == Streams.NONE) ? seed : f.apply(t);}};return StreamSupport.stream(Spliterators.spliteratorUnknownSize(iterator,Spliterator.ORDERED | Spliterator.IMMUTABLE), false);
}public static<T> Stream<T> generate(Supplier<T> s) {Objects.requireNonNull(s);return StreamSupport.stream(new StreamSpliterators.InfiniteSupplyingSpliterator.OfRef<>(Long.MAX_VALUE, s), false);
}

通过源码可以看出,iterate()方法主要是使用“迭代”的方式生成无限流,而generate()方法主要是使用“生成”的方式生成无限流。我们可以使用下面的代码示例来使用这两个方法生成Stream流。

  • 迭代

Stream<Integer> intStream = Stream.iterate(0, (x) -> x + 2);
intStream.forEach(System.out::println);

运行上述代码,会在终端一直输出偶数,这种操作会一直持续下去。如果我们只需要输出10个偶数,该如何操作呢?其实也很简单,使用Stream对象的limit方法进行限制就可以了,如下所示。

Stream<Integer> intStream = Stream.iterate(0, (x) -> x + 2);
intStream.limit(10).forEach(System.out::println);
  • 生成

Stream.generate(() -> Math.random()).forEach(System.out::println);

上述代码同样会一直输出随机数,如果我们只需要输出5个随机数,则只需要使用limit()方法进行限制即可。

Stream.generate(() -> Math.random()).limit(5).forEach(System.out::println);

(5)创建空流

在Stream类中提供了一个empty()方法,如下所示。

public static<T> Stream<T> empty() {return StreamSupport.stream(Spliterators.<T>emptySpliterator(), false);
}

我们可以使用Stream类的empty()方法来创建一个空Stream流,如下所示。

Stream<String> empty = Stream.empty();
特别推荐一个分享架构+算法的优质内容,还没关注的小伙伴,可以长按关注一下:长按订阅更多精彩▼如有收获,点个在看,诚挚感谢

【Java8新特性】面试官问我:Java8中创建Stream流有哪几种方式?相关推荐

  1. 后处理程序文件大小的变量_【每日一题】(17题)面试官问:JS中事件流,事件处理程序,事件对象的理解?...

    关注「松宝写代码」,精选好文,每日一题 作者:saucxs | songEagle 2020,实「鼠」不易 2021,「牛」转乾坤 风劲潮涌当扬帆,任重道远须奋蹄! 一.前言 2020.12.23 立 ...

  2. 面试官问我Java8~14的有哪些重要的新特性,我哭了~~~

    大家好,我是Guide哥!这篇文章来自读者的投稿,经过了两次较大的改动,两周的完善终于完成.Java 8新特性见这里:Java8新特性最佳指南 . Guide 哥:别人家的特性都用了几年了,我 Jav ...

  3. 【169期】面试官问:说说为什么要限流,有哪些解决方案?

    点击上方"Java精选",选择"设为星标" 别问别人为什么,多问自己凭什么! 下方有惊喜留言必回,有问必答! 每天 08:35 更新文章,每天进步一点点... ...

  4. 当面试官问:JS中原始类型有哪些?

    这个问题基本回答: 原始类型有:string/number/boolean/null/undefined/symbol,总共六类 其中包含了ES6新增的symbol,以及ES10新增的BigInt 回 ...

  5. 腾讯面试官问我Java中boolean类型占用多少个字节?我说一个,面试官让我回家等通知...

    来自:程序员乔戈里 ---------------第二天-------------- 什么是boolean类型,根据官方文档的描述[1]: boolean: The boolean data type ...

  6. 面试官问:mysql中时间日期类型和字符串类型的选择

    摘要:MySQL中有多种表示时间日期的数据类型,主要有YEAR.TIME.DATE.DATETIME.TIMESTAMP等 本文分享自华为云社区<一针见血,mysql中时间日期类型和字符串类型的 ...

  7. 面试官:1亿条数据批量插入 MySQL,哪种方式最快?

    这几天研究mysql优化中查询效率时,发现测试的数据太少(10万级别),利用 EXPLAIN 比较不同的 SQL 语句,不能够得到比较有效的测评数据,大多模棱两可,不敢通过这些数据下定论. 所以通过随 ...

  8. java8新特性简述

    Java8发布时间是2014年3月19日,距离今日已经很久了,那么Java8新特性你了解吗? java8是Java的一次重大升级,巨大的里程碑式的改进!! Java语言新特性: 1.与传统结合 -- ...

  9. Java8新特性Stream流详解

    陈老老老板 说明:新的专栏,本专栏专门讲Java8新特性,把平时遇到的问题与Java8的写法进行总结,需要注意的地方都标红了,一起加油. 本文是介绍Java8新特性Stream流常用方法超详细教学 说 ...

最新文章

  1. 【Android】java.lang.IllegalArgumentException Illegal character in scheme 异常
  2. Django models的诡异异常RelatedObjectDoesNotExist
  3. GitHub防黑客新措施:弃用账密验证Git操作,改用token或SSH密钥,今晚0点执行
  4. 重磅 | 19 页花书精髓笔记!你可能正需要这份知识清单
  5. php中的parse_ini_file函数
  6. 阿里:马云从未转让和退出淘宝股份 也没有这个打算
  7. 在LINUX 下安装 Realtek 8110SC 千兆网卡驱动
  8. Redis学习第八课:Redis高级实用特性(一)
  9. spark分区连接mysql_Spark数据存储和分区操作
  10. JavaScript高级程序设计笔记 事件冒泡和事件捕获
  11. 各种数字显示屏接口:LVDS, DVI, HDMI, DisplayPort, DSI
  12. latex表格内容上下居中_Latex-表格内容垂直居中
  13. Java 小Q 世界上最遥远的距离 解法二
  14. SourceInsight4.0黑色背景主题
  15. 【软件工程】需求规格说明书
  16. 印象笔记无法连接服务器解决办法
  17. 使用七牛的sdk上传报错:incorrect region
  18. edt嵌入式确定性测试_CallSerially EDT和InvokeAndBlock(第1部分)
  19. 首届波卡黑客松项目「Manta Network」的进击之路
  20. Unity3d 网页插件BestHttp使用介绍

热门文章

  1. linux java 栈_关于Java中栈与堆的思考
  2. PTA基础编程题目集-6-7 统计某类完全平方数
  3. P3902 递增(LIS+树状数组)
  4. python3里的pillow怎么安装_“python安装pillow教程“python3.4怎么安装pil
  5. html input type=quot;filequot;,科技常识:关于type=quot;filequot;的input框样式修改小结...
  6. 入门级Mat (java版)
  7. deepnode处理过的图片_这款实用的图片软件,其功能相当于十几款图片处理软件的功能之和...
  8. 如何安装python虚拟环境_如何安装python3.9以及python虚拟环境?
  9. 管理-Tomcat和Resin如何配置对指定后缀文件(如:.pptx)下载支持
  10. 第 138 章 Spark