【Stream 流】通过一个例子看遍所有Stream API使用场景

前言

上篇文章记录了方法引用,Lambda表达式等基础的知识点,这篇文章主要结合课设项目详细介绍Stream 流的API以及它的主要场景。

在这里插入图片描述

Stream API作用

在Java 8及其以后的版本中,Stream API为处理集合数据提供了强大而灵活的功能。有了Stream API,我们可以以一种声明式的方式对数据进行过滤、映射、排序、聚合等众多操作。

Stream API使用

话不多说,接下来看一下我们毕设中的一个场景。我们现在有一个电商平台的订单系统,其中有一个Order类,它包含订单号、商品名称、价格和下单时间等属性。现在,我们有一个订单列表,并希望对这个列表进行各种操作。

Order类都有这些属性和方法:

import java.time.LocalDateTime;  
import java.util.*;  
  
class Order {  
    private String orderId;  
    private String productName;  
    private double price;  
    private LocalDateTime orderTime;  
  
    // 构造方法、getter和setter这里省略  
    //重写equals方法
    @Override  
    public boolean equals(Object o) {  
        if (this == o) return true;  
        if (o == null || getClass() != o.getClass()) return false;  
        Order order = (Order) o;  
        return Double.compare(order.price, price) == 0 &&  
                Objects.equals(orderId, order.orderId) &&  
                Objects.equals(productName, order.productName) &&  
                Objects.equals(orderTime, order.orderTime);  
    }  
  //重写HashCode方法
    @Override  
    public int hashCode() {  
        return Objects.hash(orderId, productName, price, orderTime);  
    }  
  //重写toString方法
    @Override  
    public String toString() {  
        return "Order{" +  
                "orderId='" + orderId + '\'' +  
                ", productName='" + productName + '\'' +  
                ", price=" + price +  
                ", orderTime=" + orderTime +  
                '}';  
    }  
}

有了属性和方法的类之后,我们就可以创建一个订单列表,润润喉来演示Stream API的多个方法了。

过滤(使用filter方法)

需求:找出价格大于100的订单:

List<Order> orders =  ...   //这个不重要,省略
List<Order> expensiveOrders = orders.stream()  
    .filter(order -> order.getPrice() > 100)  
    .collect(Collectors.toList());

映射(使用map方法)

需求:提取所有订单的商品名称,这里使用到了方法引用,引用Order类中的getProductName方法:

List<String> productNames = orders.stream()  
    .map(Order::getProductName)  
    .collect(Collectors.toList());

排序(使用sorted方法)

需求:按照价格对订单进行排序:

List<Order> sortedOrders = orders.stream()  
    .sorted(Comparator.comparingDouble(Order::getPrice))  
    .collect(Collectors.toList());

去重(使用distinct方法)

需求:订单列表中有重复订单(订单号相同代表着重复),我们去重后可以得到唯一的订单列表:

List<Order> uniqueOrders = orders.stream()  
    .distinct()  
    .collect(Collectors.toList());

这里需要注意:使用distinct方法去重,Order类必须正确实现equals和hashCode方法,在咱们上面创建的类中可查看详情。

计数(使用count方法)

需求:统计订单列表中订单的数量:

long orderCount = orders.stream()  
    .count();

聚合(使用reduce方法)

需求:计算订单列表中所有订单的总价格:

double totalPrice = orders.stream()  
    .mapToDouble(Order::getPrice)  
    .sum();

或使用reduce方法:

OptionalDouble totalPriceOptional = orders.stream()  
    .mapToDouble(Order::getPrice)  
    .reduce(Double::sum);  
double totalPrice = totalPriceOptional.orElse(0);

任意匹配和所有匹配(使用anyMatch和allMatch方法)

需求:检查是否有价格超过200的订单:

boolean hasExpensiveOrder = orders.stream()  
    .anyMatch(order -> order.getPrice() > 200);

检查所有订单的价格是否都大于0:

boolean areAllPricesValid = orders.stream()  
    .allMatch(order -> order.getPrice() > 0);

查找第一个/最后一个元素(使用findFirst和findAny方法)

需求:查找价格最高的订单(假设订单已按价格排序):

Optional<Order> mostExpensiveOrder = orders.stream()  
    .sorted(Comparator.comparingDouble(Order::getPrice).reversed())  
    .findFirst();

这里需要注意:在没有排序的情况下,findFirst返回的是流中的第一个元素,而findAny返回流中的任意元素。在并行流中,findAny通常比findFirst更高效。

扁平化流(使用flatMap方法)

需求:每个订单有一个商品列表,我们想要得到一个包含所有商品名称的流:

class Order {  
    //  属性和方法  
    List<String> productNames; // 假如每个订单有这个属性  
      
    // getter和setter  
}  
  
// 初始化orders列表这里也省略
  
Stream<String> allProductNames = orders.stream()  
    .flatMap(order -> order.getProductNames().stream());  
  
List<String> collectedProductNames = allProductNames.collect(Collectors.toList());

分组(使用collect方法和Collectors.groupingBy)

需求:按照商品价格区间对订单进行分组:

Map<String, List<Order>> groupedOrders = orders.stream()  
    .collect(Collectors.groupingBy(  
        order -> {  
            if (order.getPrice() <= 100) {  
                return "Cheap";  
            } else if (order.getPrice() <= 200) {  
                return "Medium";  
            } else {  
                return "Expensive";  
            }  
        }  
    ));

分区(使用collect方法和Collectors.partitioningBy)

需求:将订单按照价格是否超过100进行分区:

Map<Boolean, List<Order>> partitionedOrders = orders.stream()  
    .collect(Collectors.partitioningBy(order -> order.getPrice() > 100));

连接字符串(使用collect方法和Collectors.joining)

需求:将所有订单的商品名称连接成一个字符串,以逗号分隔:

String productNamesString = orders.stream()  
    .flatMap(order -> order.getProductNames().stream())  
    .collect(Collectors.joining(", "));

使用自定义收集器

需求:我们想要收集所有订单的价格到一个IntSummaryStatistics对象中,以便获取最大值、最小值、平均值等统计信息:

IntSummaryStatistics priceStats = orders.stream()  
    .mapToInt(Order::getPrice)  
    .summaryStatistics();  
  
System.out.println("Max price: " + priceStats.getMax());  
System.out.println("Min price: " + priceStats.getMin());  
System.out.println("Average price: " + priceStats.getAverage());

限制流的大小(使用limit方法)

需求:获取价格最高的前三个订单:

List<Order> top3ExpensiveOrders = orders.stream()  
    .sorted(Comparator.comparingDouble(Order::getPrice).reversed())  
    .limit(3)  
    .collect(Collectors.toList());

跳过流中的元素(使用skip方法)

需求:跳过前五个订单,然后获取剩余的订单:

List<Order> remainingOrders = orders.stream()  
    .skip(5)  
    .collect(Collectors.toList());

这些场景涵盖了Stream API中的大部分常用方法。咱们列举的也不少了,今天就列举到这里吧。

本篇文章到此结束,谢谢大家的观看!

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/597709.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Petalinux的使用——定制Linux系统

文章目录 配置petalinux运行环境petalinux设计流程 配置petalinux运行环境 Petalinux的安装在文章Ubuntu镜像源的更改及其Petalinux的安装中已经介绍&#xff0c;下面介绍petalinux运行环境的配置过程。 进入到petalinux的安装路径下&#xff0c;使用下面的命令对petalinux的运…

类和对象-Python-第一部分

初识对象 使用对象组织数据 class Student:nameNonegenderNonenationalityNonenative_placeNoneageNonestu_1Student()stu_1.name"林军杰" stu_1.gender"男" stu_1.nationality"中国" stu_1.native_place"山东" stu_1.age31print(stu…

《视觉十四讲》例程运行记录(2)——运行ch4的例程评估轨迹误差

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 一、运行ch4的example1. 编译例程2. 运行报错(1) 报错一(2) 报错二 一、运行ch4的example 1. 编译例程 (1) 在slambook2/ch4/example目录下&#xff0c;创建build…

五月节放假作业讲解

目录 作业1&#xff1a; 问题&#xff1a; 结果如下 作业2&#xff1a; 结果: 作业1&#xff1a; 初始化数组 问题&#xff1a; 如果让数组初始化非0数会有问题 有同学就问了&#xff0c;我明明已经初始化定义过了&#xff0c;为啥还有0呀 其实这种初始化只会改变第一个…

【嵌入式笔试题】网络编程笔试题

非常经典的笔试题。 2.网络编程(29道) 2.1列举一下OSI协议的各种分层。说说你最熟悉的一层协议的功能。 ( 1 )七层划分为:应用层、表示层、会话层、传输层、网络层、数据链路层、物理 层。 ( 2 )五层划分为:应用层、传输层、网络层、数据链路层、物理层。 ( 3 )…

视频改字祝福/豪车装X系统源码/小程序uniapp前端源码

uniapp视频改字祝福小程序源码&#xff0c;全开源。创意无限&#xff01;AI视频改字祝福&#xff0c;豪车装X系统源码开源&#xff0c;打造个性化祝福视频不再难&#xff01; 想要为你的朋友或家人送上一份特别的祝福&#xff0c;让他们感受到你的真诚与关怀吗&#xff1f;现在…

VALSE 2024年度进展评述内容分享-视觉基础大模型的进展

2024年视觉与学习青年学者研讨会&#xff08;VALSE 2024&#xff09;于5月5日到7日在重庆悦来国际会议中心举行。本公众号将全方位地对会议的热点进行报道&#xff0c;方便广大读者跟踪和了解人工智能的前沿理论和技术。欢迎广大读者对文章进行关注、阅读和转发。文章是对报告人…

探无止境,云游未来 | “游戏出海云”发布

4月28日下午&#xff0c;2024中国移动算力网络大会之“游戏出海”分论坛在江苏省苏州金鸡湖国际会议中心圆满落幕。 此次论坛由中国移动海南公司主办&#xff0c;中国移动通信集团政企事业部、中国移动云能力中心、中国移动国际公司共同协办。海南省工业与信息化厅副厅长黄业晶…

【栈】Leetcode 1047. 删除字符串中的所有相邻重复项

题目讲解 1047. 删除字符串中的所有相邻重复项 算法讲解 使用栈这个数据结构&#xff0c;每一次入栈的时候观察此时的字符和当前栈顶字符是否相等&#xff0c;如相等&#xff1a;栈顶出栈&#xff1b;不相等&#xff1a;入栈 class Solution { public:string removeDuplica…

Linux 基础IO篇

1. C语言中的文件操作 1 #include<stdio.h>2 3 int main()4 {5 FILE* pf fopen("log.txt", "w");6 if(NULL pf)7 {8 perror("fopen");9 return 1; 10 }11 fprintf(pf, &quo…

Spring框架学习笔记(一):Spring基本介绍(包含容器底层结构)

1 官方资料 1.1 官网 https://spring.io/ 1.2 进入 Spring5 下拉 projects, 进入 Spring Framework 进入 Spring5 的 github 1.3 在maven项目中导入依赖 <dependencies><!--加入spring开发的基本包--><dependency><groupId>org.springframework<…

ICode国际青少年编程竞赛- Python-1级训练场-变量的计算

ICode国际青少年编程竞赛- Python-1级训练场-变量的计算 1、 a 2 for i in range(4):Spaceship.step(a-1)Dev.step(a)Dev.step(-a)a a 12、 a 2 for i in range(4):Dev.step(2 a)Dev.step(-a)Dev.turnRight()a a 13、 y 4 for i in range(3):Dev.step(y)Dev.turnRigh…

数据同步新突破!一招解决文化公司系统对接难题!

一、客户介绍 某文化传播有限公司&#xff0c;是一家专注于文化艺术领域&#xff0c;集创作、制作、交流等多功能于一体的公司。公司始终秉承创意和质量的双重标准&#xff0c;为观众带来一系列高质量的文化艺术作品。该公司的经营范围广泛&#xff0c;涵盖了组织文化艺术交流…

cmake进阶:变量的作用域说明三(从函数作用域方面)

一. 简介 前一篇文章从函数作用域方面学习了 变量的作用域。文章如下&#xff1a; cmake进阶&#xff1a;变量的作用域说明一&#xff08;从函数作用域方面&#xff09;-CSDN博客cmake进阶&#xff1a;变量的作用域说明二&#xff08;从函数作用域方面&#xff09;-CSDN博客…

VALSE 2024年度进展评述内容分享-视觉通用人工智能

2024年视觉与学习青年学者研讨会&#xff08;VALSE 2024&#xff09;于5月5日到7日在重庆悦来国际会议中心举行。本公众号将全方位地对会议的热点进行报道&#xff0c;方便广大读者跟踪和了解人工智能的前沿理论和技术。欢迎广大读者对文章进行关注、阅读和转发。文章是对报告人…

优雅处理返回信息状态码:Result对象在Spring Boot中的应用

前言 在开发过程中&#xff0c;处理返回的信息状态码是一个重要的问题&#xff0c;尤其是在大型项目中。为了统一处理这些状态码&#xff0c;我在Spring Boot中创建了一个名为Result的Java对象&#xff0c;用于封装返回的信息和状态码。在本文中&#xff0c;我将分享如何实现这…

【C++题解】1435. 数池塘(八方向)

问题&#xff1a;1435. 数池塘&#xff08;八方向&#xff09; 类型&#xff1a;深搜 题目描述&#xff1a; 农夫约翰的农场可以表示成 NM&#xff08;1≤N,M≤100&#xff09;个方格组成的矩形。由于近日的降雨&#xff0c;在约翰农场上的不同地方形成了池塘。 每一个方格或…

重写muduo之获取线程tid代码

目录 1、概述 2、CurrentThread.h 3、 CurrentThread.cc 1、概述 我们的服务器程序不一定就只有1个Eventloop&#xff0c;我们可能有很多的Eventloop&#xff0c;每个Eventloop都有很多channel&#xff0c;自己channel上的事件要在自己的Eventloop线程上去处理&#xff0c;E…

免费开源的tiktok加速软件

背景 tiktok是国内企业出海做的比较成功的案例&#xff0c;可以简单的理解为海外版的抖音&#xff0c;关于tiktok的运营和变现不是我们擅长的领域&#xff0c;这里就不多说了&#xff0c;这篇文章主要着重在解决tiktok的网络问题&#xff0c;介绍如何用开源的软件自己搭建一套…

基于 Ubuntu22.04 安装 SSH 服务

文章目录 一、Ubuntu22.04 安装 SSH 服务二、配置 OpenSSH&#xff08;安全性&#xff09;1. 更改 OpenSSH 端口2. 限制使用 SSH 登录尝试次数3. 禁止 SSH 以 root 身份连接 三、设置防火墙&#xff08;UFW&#xff09;锁定 SSH四、远程终端软件通过 SSH 连接 Ubuntu22.041. 远…