转载自   跟我学 Java 8 新特性之 Stream 流(七)流与迭代器,流系列大结局

恭喜你们,马上就要学完Java8 Stream流的一整系列了,其实我相信Stream流对很多使用Java的同学来说,都是一个知识盲点,因为这个原因,我才这么细致地讲解Stream流的各个知识点,通过这一整个系列,我相信只要认真看了的同学,都已掌握的差不多了,就差实战了。

其实我在工作过程中,Stream流对我的帮助真的挺大的,所以,我想和大家分享一下,于是这系列的文章就出来了。

在本系列文章发布的时候,有很多同学反映,Stream流的调试和forEach()的调试都不是特别友好,那本篇给出一个折中的调试方法,虽然不能完美解决调试的问题,但是基本上已经能解决绝大部分的调试问题了,没错,就是迭代器了,当然迭代器除了能辅助调试以外,他最重要的还是遍历功能。

这篇文章除了介绍传统的迭代器外,还会介绍Java8中新增的Spliterator,因为在并行流的场合,Spliterator相当的好用。

先简单介绍一下传统的迭代器

迭代器是实现了Iterator接口的对象,并且Iterator接口允许遍历,获取或者移除元素。

public interface Iterator<E> {Iterator<T> iterator();E next();default void remove() {throw new UnsupportedOperationException("remove");}default void forEachRemaining(Consumer<? super E> action) {Objects.requireNonNull(action);while (hasNext())action.accept(next());}}

使用Iterator的一般步骤:

(1)通过iterator()方法,获取指向集合或流开头的迭代器。

(2)建立一个hasNext()方法调用循环,只要hasNext()返回true,就继续迭代。

(3)在循环中,通过调用next()方法获取每个元素。

private static void learnIterator() {List<String> lists = Arrays.asList("A", "B", "C", "D");Iterator<String> iterator = lists.stream().iterator();while (iterator.hasNext()) {System.out.println(iterator.next());}
}

但是如果我们不修改集合的情况下,使用forEach()其实更加便利的,其实两种方式本质上面是一样的,在你编译之后,forEach()会转换成迭代器的方式进行操作了。有了迭代器,相信调试就得方便起来了,即使不能直接调试,也可以通过迭代器,反推之前,可能发生了什么。

值得注意的一点是:Java8给基本类型的流提供了额外的迭代器:PrimitiveIterator.OfInt,PrimitiveIterator.OfDouble,PrimitiveIterator.OfLong,PrimitiveIterator,但这些接口都是来扩展自Iterator接口的,所以使用上面也是相同的。

Spliterator

Spliterator是Java8新增的一种迭代器,这种迭代器由Spliterator接口定义,Spliterator也有普通的遍历元素功能,这一点与刚才说的迭代器类似的,但是,但是Spliterator方法和使用迭代器的方法是不同的。

另外,它提供的功能要比Iterator多。最终要的一点,Spliterator支持并行迭代。

public interface Spliterator<T> {boolean tryAdvance(Consumer<? super T> action);int characteristics();long estimateSize();Spliterator<T> trySplit();default void forEachRemaining(Consumer<? super T> action) {do { } while (tryAdvance(action));}
}

将Spliterator用于基本迭代任务是非常简单的,只需要调用tryAdvance()方法,直至其返回false.如果要为序列中的每个元素应用相同的动作,那么forEachRemaining()提供了一种更加高效的替代方法。

对于这两个方法,在每次迭代中将发生的动作都由Consumer对象定义的操作来决定,Consumer也是一个函数式接口,估计大家已经知道怎么分析了,这里就不带大家分析了,他的动作是指定了在迭代中下一个元素上执行的操作。下面来一个简单的例子:

private static void learnIterator() {List<String> lists = Arrays.asList("A", "B", "C", "D");Spliterator<String> spliterator = lists.stream().spliterator();while (spliterator.tryAdvance(System.out::println));
}

使用forEachRemaining()方法改进这个例子:

private static void learnIterator() {List<String> lists = Arrays.asList("A", "B", "C", "D");lists.stream().spliterator().forEachRemaining(System.out::println);
}

注意,使用这个方法时,不需要提供一个循环来一次处理一个元素,而是将各个元素作为一个整体来对待,这是Spliterator的又一个优势。

Spliterator的另一个值得注意的方法是trySplit(),它将被迭代的元素划分成了两部分,返回其中一部分的新Spliterator,另一部分则通过原来的Spliterator访问。下面再给一个简单的例子

private static void learnIterator() {List<String> lists = Arrays.asList("A", "B", "C", "D");Spliterator<String> spliterator = lists.stream().spliterator();Spliterator<String> stringSpliterator = spliterator.trySplit();if (stringSpliterator != null) stringSpliterator.forEachRemaining(System.out::println);System.out.println("------------------");spliterator.forEachRemaining(System.out::println);
}

打印的结果:

A
B
------------------
C
D

这里只是给大家提供了这种方式而已,例子本身没有什么含义,但是当你对大数据集执行并行处理时,拆分可能是极有帮助的了。但更多情况下,要对流执行并行操作时,使用其他某个Stream方法更好,而不必手动处理Spliterator的这些细节,Spliterator最适合的场景是,给定的所有方法都不能满足你的要求时,才考虑。

最后来一个总结

到这里,Java8 Stream流的知识,基本上已经介绍完了,缩减操作,并行流,映射,还有收集是Stream流的核心内容,但是要想掌握好这些内容,第二篇的基本知识你是跑不掉的,因此我的建议还是老老实实,从第一篇开始认真看,看完这7篇文章,你基本已经非常系统地掌握Java8的Stream流知识了。

大家也不要忘记Collectors类,里面提供给我们的方法,基本上能处理各种各样的收集元素问题了,如果你从第一篇文章开始看,一直看到这里了,那你以后尽量多使用Stream流来处理集合吧,只是学会这些知识是没有用的,要多在你们的项目中运用他们,这样才能更加好地理解,如果大家在使有的过程中遇到什么样的问题,欢迎来这里面留言,我看到了,一定第一时间和大家一起探索解决的方法。

如果大家对这一系列文章感兴趣的话,也可以关注一下本公众号,后面作者会出一系列数据结构方面的文章,当然是Java的一整个系列数据结构啦。

如果你觉得Stream流这系列文章,你们能看得比较懂的话,那我相信数据结构系列的文章你们也会有同样的感觉的,最后,谢谢大家,祝你们技术越来越强,生活越来越愉快,最重要的还是马上找到女朋友,哈哈哈哈。

跟我学 Java 8 新特性之 Stream 流(七)流与迭代器,流系列大结局相关推荐

  1. 跟我学 Java 8 新特性之 Stream 流(六)收集

    转载自   跟我学 Java 8 新特性之 Stream 流(六)收集 我们前面的五篇文章基本都是在说将一个集合转成一个流,然后对流进行操作,其实这种操作是最多的,但有时候我们也是需要从流中收集起一些 ...

  2. 跟我学 Java 8 新特性之 Stream 流基础体验

    转载自   跟我学 Java 8 新特性之 Stream 流基础体验 Java8新增的功能中,要数lambda表达式和流API最为重要了.这篇文章主要介绍流API的基础,也是流API系列的第一篇文章, ...

  3. 跟我学 Java 8 新特性之 Stream 流(二)关键知识点

    转载自   跟我学 Java 8 新特性之 Stream 流(二)关键知识点 我们的第一篇文章,主要是通过一个Demo,让大家体验了一下使用流API的那种酣畅淋漓的感觉.如果你没有实践,我还是再次呼吁 ...

  4. 跟我学 Java 8 新特性之 Stream 流(三)缩减操作

    转载自   跟我学 Java 8 新特性之 Stream 流(三)缩减操作 和前面两篇文章一起服用,效果会更佳.通过对流API的基础体验Demo和关键知识点的讲解,相信大家对流API都有一定的认识了, ...

  5. 跟我学 Java 8 新特性之 Stream 流(四)并行流

    转载自   跟我学 Java 8 新特性之 Stream 流(四)并行流 随着对流API认识的慢慢深入,本章我们要讨论的知识点是流API里面的并行流了. 在开始讨论并行流之前,我先引发一下大家的思考, ...

  6. 跟我学 Java 8 新特性之 Stream 流(五)映射

    转载自   跟我学 Java 8 新特性之 Stream 流(五)映射 经过了前面四篇文章的学习,相信大家对Stream流已经是相当的熟悉了,同时也掌握了一些高级功能了,如果你之前有阅读过集合框架的基 ...

  7. Java 8 新特性之Stream API

    1. 概述 1.1 简介 Java 8 中有两大最为重要的改革,第一个是 Lambda 表达式,另外一个则是 Stream API(java.util.stream.*). Stream 是 Java ...

  8. Java 8 新特性之 Stream 流(五)映射

    经过了前面四篇文章的学习,相信大家对Stream流已经是相当的熟悉了,同时也掌握了一些高级功能了,如果你之前有阅读过集合框架的基石Collection接口,是不是在经过前面的学习,以前看不懂的东西,突 ...

  9. Java 8 新特性 宋红康跟学

    Java 8 新特性 Java 8 新特性简介 思维导图 并行流与串行流 Lambda表达式 Lambda 表达式语法 语法格式一:无参,无返回值,Lambda体只需一条语句 语法格式二:Lambda ...

最新文章

  1. Oracle10g.CentOS6安装
  2. 通过ssh建立点对点的隧道,实现两个子网通信
  3. python 中的转义序列
  4. 爱丁堡大学人工智能和计算机科学研究生申请入口
  5. Win切换管理员用户
  6. java ee api_Java EE并发API教程
  7. IBatisNet XML 特殊配置
  8. appium学习【三】:截图时,图片命令中包含当前的函数名,以区分错误是在哪个函数报的...
  9. 网课查课插件 支持 60类型+ 彩虹查课插件 网络查课查询
  10. weblogic启动项目失败查看_weblogic 部署项目成功,但是再界面上取不到数据
  11. android的wifi开发,android开发教程之wifi开发示例
  12. 基于AT89S52的俄罗斯方块游戏设计与实现
  13. 笔记本电脑触摸板手势命令
  14. win7系统文件夹共享后有锁图标怎么去掉?
  15. 从-Quora-的-187-个问题中学习机器学习和NLP
  16. 神经网络的三种训练方法,神经网络训练过程详解
  17. 如何提升APP日活(DAU)?
  18. 3款最伟大的联机游戏
  19. java 虚拟机 Java内存结构 JVM垃圾回收机制算法
  20. 【T-SQL】一、数据库的创建与管理

热门文章

  1. BS作业 基于springboot + Thymeleaf +mybatis 实现的书城管理系统
  2. java 视频监控 分屏ui_视频监控网页ActiveX视频分屏播放控件开发
  3. java堆和栈 常量池_GitHub - han-guang-xue/difference-of-stack-heap-pool: Java中堆、栈和常量池的区别...
  4. [JavaWeb-Bootstrap]Bootstrap快速入门
  5. 支付价格计算中精度问题之double,float
  6. sketch里的ios控件_30个让你眼前一亮的iOS Swift UI控件!
  7. C语言---二进制和文本文件的备份
  8. 【LOJ#6682】梦中的数论(min_25筛)
  9. Codeforces Round #619 (Div. 2) D. Time to Run 矩阵回路构造
  10. 牛客 CCA的区间 dp + 补集转移