Java8新特性整理之流的介绍与使用(三)

2024-06-16 05:58

本文主要是介绍Java8新特性整理之流的介绍与使用(三),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

流是什么

官方定义:支持顺序和并行聚合操作的元素序列。

这里有几个关键词,顺序、并行、聚合、元素序列。

所谓顺序就是单线程顺序执行,并行就是多线程分解执行,聚合就是将顺序或并行执行的结果计算后得出最终结果,元素序列则是将数据源(数组,文件,集合等)流化后的数据结构。

流与集合

上面说的还是有些不明朗,下面结合Java中的集合(Collection)来进一步解释流。

Java现有的集合概念和新的流概念都提供了接口,来配合代表元素型有序值的数据接口。所
谓有序,就是说我们一般是按顺序取用值,而不是随机取用的。

举例来说,如果要观看苍老师课程,有两种方式,一种是下载到本地观看,另一种是在线观看。而下载观看的方式等待下载的时间比较长同时占用磁盘空间较大;在线观看就比较快了,可以只观看高潮部分,而且只占用很少的缓冲空间。

对比来看,苍老师课程是字节或帧的数据结构,用集合方式处理的话(下载到本地观看)需要将整个结构中的数据都计算一遍,而用流的方式处理的话(在线观看高潮部分),只需要计算某个字节(帧)范围。

区别

  • 集合是内存数据结构,可以增删元素,而流是概念上固定的数据结构,不可以增删元素,只进行计算。
  • 集合可以多次遍历,而流只能遍历一次(下一次需要从数据源再获得一个新的流)。
  • 集合使用外部迭代(如for-each),而流使用内部迭代(流内部帮你把迭代做了)。
  • 关键区别,集合是有界的,流可以是无界的。

流的操作

java.util.stream.Stream中的Stream接口定义了许多操作。它们可以分为两大类。

  • 中间操作,诸如filter或sorted等中间操作会返回另一个流,即返回值为Stream的方法。
  • 终端操作,终端操作会从流的流水线生成结果,其结果是任何不是流的值,即返回值不为Stream的方法。

使用流

流的使用一般包括三件事:
- 一个数据源(如集合)来执行一个查询;
- 一个中间操作链,形成一条流的流水线;
- 一个终端操作,执行流水线,并能生成结果。

下面介绍下流中的常用方法:

中间操作常用方法

filter方法:

接受一个返回boolean的Lambda表达式的参数,返回由流元素组成的流,该流与给定谓词匹配。

distinct方法:

返回由不同对象组成的流,内部使用对象的equals方法比较是否相同。

skip方法:

接受一个long类型的参数n,表示头n个数,返回由剩下的元素组成的流,如果流容器中的元素比n小,则返回空的流。

limit方法:

接受一个long类型的参数maxSize,表示限制的最大数量,返回一个不超过maxSize长度的流。

sorted方法:

接受一个Comparator类型的参数,表示函数引用作为参数,返回根据Comparator接口中定义的行为组成排序后的流。

map方法:

接收一个函数(方法引用)作为参数,返回一个流,由将给定函数应用于该流元素的结果组成。

终端操作常用方法

anyMatch方法:

流中是否有一个元素能匹配给定的谓词

allMatch方法:

流中的元素是否都能匹配给定的谓词。

noneMatch方法:

和allMatch相对的是noneMatch。它可以确保流中没有任何元素与给定的谓词匹配。

findAny方法:

返回当前流中的任意元素。

findFirst方法:

找到第一个元素。

forEach方法:

遍历流中的每一个元素。

collect方法:

接受一个Collectors类中的方法(收集器)作为参数,返回一个归约的结果。

collect是一个终端操作,它接受的参数是将流中元素累积到汇总结果的各种方式(称为收集器)。

reduce方法:

把一个流中的元素反复结合起来,返回一个归约的结果(将流归约成一个值)。

count方法:

返回流中的元素个数。

举个例子

前面都是理论知识,下面举个栗子:

Dish.java

public class Dish {private final String name;private final boolean vegetarian; // 是否是素食private final int calories; // 卡路里private final Type type;  // 盘子装的菜的类型public Dish(String name, boolean vegetarian, int calories, Type type) {this.name = name;this.vegetarian = vegetarian;this.calories = calories;this.type = type;}public String getName() {return name;}public boolean isVegetarian() {return vegetarian;}public int getCalories() {return calories;}public Type getType() {return type;}public enum Type {MEAT, FISH, OTHER;}
}

初始化数据:

List<Dish> menu = Arrays.asList(new Dish("pork", false, 800, Dish.Type.MEAT),new Dish("beef", false, 700, Dish.Type.MEAT),new Dish("chicken", false, 400, Dish.Type.MEAT),new Dish("french fries", true, 530, Dish.Type.OTHER),new Dish("rice", true, 350, Dish.Type.OTHER),new Dish("season fruit", true, 120, Dish.Type.OTHER),new Dish("pizza", true, 550, Dish.Type.OTHER),new Dish("prawns", false, 300, Dish.Type.FISH),new Dish("salmon", false, 450, Dish.Type.FISH));

使用:

List<String> dishes = menu.stream().filter(dish -> dish.getCalories() > 300)   // 从流中过滤元素.map(Dish::getName)                         // 提取元素.limit(3)                                   // 截断流,使其元素不超过给定的数量.collect(toList());                         // 将流转换为列表  

上面例子会取出卡路里大于300的前三个Dish的名字列表。

数值流

下面来谈谈流的特化 – 数值流

对前面Dish中的菜的卡路里求和:

int calories = menu.stream().map(Dish::getCalories).reduce(0, Integer::sum);

reduce方法第一个参数表示初始值,第二个参数代表接受两个参数的函数,Integer的sum方法接受两个参数,所以可以传递一个方法引用。

乍一看,这个方法好像没什么问题,输出结果也正确。

但你其实忽略了一个问题,map方法会返回一个Stream类型的流,其中T是引用类型,所以它有一个暗含的装箱成本,会造成性能的降低。

怎么解决上面的问题呢?

Java 8引入了三个原始类型特化流接口来解决这个问题:IntStream、DoubleStream和LongStream,分别将流中的元素特化为int、long和double,从而避免了暗含的装箱成本。每个接口都带来了进行常用数值归约的新方法,比如对数值流求和的sum,找到最大元素的max。

现在改正上面的代码:

int calories = menu.stream().mapToInt(Dish::getCalories).sum();

当然,如有你需要转换回对象流,则需要调用原始类型特化流接口的boxed方法进行装箱。

什么是并行流

并行流就是一个把内容分成多个数据块,并用不同的线程分别处理每个数据块的流。

这样一来,你就可以自动把给定操作的工作负荷分配给多核处理器的所有内核,让它们都忙起来。

可以通过对收集源调用parallelStreamparallel方法来把集合转换为并行流:

 int calories = menu.parallelStream().mapToInt(Dish::getCalories).sum();

 int calories = menu.stream().parallel().mapToInt(Dish::getCalories).sum();

下表按照可分解性总结了一些流数据源适不适于并行。

可分解性
ArrayList极佳
LinkedList
IntStream.range极佳
Stream.iterate
HashSet
TreeSet

这篇关于Java8新特性整理之流的介绍与使用(三)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1065637

相关文章

详解Vue如何使用xlsx库导出Excel文件

《详解Vue如何使用xlsx库导出Excel文件》第三方库xlsx提供了强大的功能来处理Excel文件,它可以简化导出Excel文件这个过程,本文将为大家详细介绍一下它的具体使用,需要的小伙伴可以了解... 目录1. 安装依赖2. 创建vue组件3. 解释代码在Vue.js项目中导出Excel文件,使用第三

Linux alias的三种使用场景方式

《Linuxalias的三种使用场景方式》文章介绍了Linux中`alias`命令的三种使用场景:临时别名、用户级别别名和系统级别别名,临时别名仅在当前终端有效,用户级别别名在当前用户下所有终端有效... 目录linux alias三种使用场景一次性适用于当前用户全局生效,所有用户都可调用删除总结Linux

Java实现Excel与HTML互转

《Java实现Excel与HTML互转》Excel是一种电子表格格式,而HTM则是一种用于创建网页的标记语言,虽然两者在用途上存在差异,但有时我们需要将数据从一种格式转换为另一种格式,下面我们就来看看... Excel是一种电子表格格式,广泛用于数据处理和分析,而HTM则是一种用于创建网页的标记语言。虽然两

java图像识别工具类(ImageRecognitionUtils)使用实例详解

《java图像识别工具类(ImageRecognitionUtils)使用实例详解》:本文主要介绍如何在Java中使用OpenCV进行图像识别,包括图像加载、预处理、分类、人脸检测和特征提取等步骤... 目录前言1. 图像识别的背景与作用2. 设计目标3. 项目依赖4. 设计与实现 ImageRecogni

Java中Springboot集成Kafka实现消息发送和接收功能

《Java中Springboot集成Kafka实现消息发送和接收功能》Kafka是一个高吞吐量的分布式发布-订阅消息系统,主要用于处理大规模数据流,它由生产者、消费者、主题、分区和代理等组件构成,Ka... 目录一、Kafka 简介二、Kafka 功能三、POM依赖四、配置文件五、生产者六、消费者一、Kaf

Java访问修饰符public、private、protected及默认访问权限详解

《Java访问修饰符public、private、protected及默认访问权限详解》:本文主要介绍Java访问修饰符public、private、protected及默认访问权限的相关资料,每... 目录前言1. public 访问修饰符特点:示例:适用场景:2. private 访问修饰符特点:示例:

python管理工具之conda安装部署及使用详解

《python管理工具之conda安装部署及使用详解》这篇文章详细介绍了如何安装和使用conda来管理Python环境,它涵盖了从安装部署、镜像源配置到具体的conda使用方法,包括创建、激活、安装包... 目录pytpshheraerUhon管理工具:conda部署+使用一、安装部署1、 下载2、 安装3

Mysql虚拟列的使用场景

《Mysql虚拟列的使用场景》MySQL虚拟列是一种在查询时动态生成的特殊列,它不占用存储空间,可以提高查询效率和数据处理便利性,本文给大家介绍Mysql虚拟列的相关知识,感兴趣的朋友一起看看吧... 目录1. 介绍mysql虚拟列1.1 定义和作用1.2 虚拟列与普通列的区别2. MySQL虚拟列的类型2

Python进阶之Excel基本操作介绍

《Python进阶之Excel基本操作介绍》在现实中,很多工作都需要与数据打交道,Excel作为常用的数据处理工具,一直备受人们的青睐,本文主要为大家介绍了一些Python中Excel的基本操作,希望... 目录概述写入使用 xlwt使用 XlsxWriter读取修改概述在现实中,很多工作都需要与数据打交

详解Java如何向http/https接口发出请求

《详解Java如何向http/https接口发出请求》这篇文章主要为大家详细介绍了Java如何实现向http/https接口发出请求,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 用Java发送web请求所用到的包都在java.net下,在具体使用时可以用如下代码,你可以把它封装成一