探索Java Stream API:高效处理集合的利器

news2024/9/22 17:22:29

文章目录

    • 一、Stream API简介
      • 1.1 什么是Stream?
      • 1.2 Stream的特点
    • 二、Stream API的基本操作
      • 2.1 创建Stream
      • 2.2 中间操作
      • 2.3 终端操作
    • 三、Stream API的高级应用
      • 3.1 并行Stream
      • 3.2 复杂数据处理
      • 3.3 Stream与Optional
    • 四、最佳实践
      • 例子 1: 筛选和映射
      • 例子 2: 排序和收集
      • 例子 3: 分组和汇总
      • 例子 4: 并行流
    • 五、总结

在Java 8及之后的版本中,Stream API的引入无疑为Java的集合处理带来了革命性的变化。Stream API提供了一种高效、声明式的方式来处理数据集合(如List、Set等),使得数据处理代码更加简洁、易于理解和维护。本文将深入探讨Java Stream API的核心概念、常用操作、以及如何通过Stream实现复杂的数据处理逻辑。

在这里插入图片描述

一、Stream API简介

1.1 什么是Stream?

Stream(流)是Java 8中引入的一个关键抽象概念,它允许你以声明方式处理数据集合(包括数组)。Stream API通过对集合(Collection)的封装,提供了一种高级迭代器抽象。通过Stream,你可以利用Lambda表达式来执行复杂的集合操作,如筛选、排序、映射等,而无需修改原始数据源。

1.2 Stream的特点

非破坏性:Stream操作不会修改原始数据源。
惰性求值:Stream操作是延迟执行的,只有在需要结果时才会进行实际处理。
可消费性:Stream只能被消费一次,一旦遍历过,就不能再次遍历。
中间操作与终端操作:Stream操作分为中间操作和终端操作,中间操作返回Stream本身,可以链式调用;终端操作则返回一个结果或副作用,如集合、void等。

二、Stream API的基本操作

2.1 创建Stream

通过集合的.stream()或.parallelStream()方法创建。
通过Stream.of()、Stream.builder()或Stream.generate()等静态方法创建。
通过数组的Arrays.stream()方法创建。

2.2 中间操作

筛选(filter):通过Lambda表达式过滤元素。
映射(map):将每个元素映射(转换)成另一种形式。
排序(sorted):对流中的元素进行排序。
去重(distinct):去除流中的重复元素。
扁平化(flatMap):将流中的每个元素转换成流,然后将所有流连接成一个流。

2.3 终端操作

收集(collect):将流中的元素收集到新的集合中。
匹配(anyMatch、allMatch、noneMatch):检查流中的元素是否满足某种条件。
归约(reduce):将流中的元素反复结合起来,得到一个值。
查找(findFirst、findAny):查找流中的元素。
forEach:对流中的每个元素执行操作。

三、Stream API的高级应用

3.1 并行Stream

并行Stream利用多核处理器的优势,可以同时处理流中的多个元素,显著提高处理速度。通过集合的.parallelStream()方法可以获取并行Stream。但需要注意的是,并非所有情况下并行处理都能带来性能提升,合理的线程管理和数据分割是关键。

3.2 复杂数据处理

Stream API结合Lambda表达式和函数式接口,可以非常灵活地处理复杂的数据转换和聚合逻辑。例如,可以通过Collectors类提供的静态方法来实现复杂的收集操作,如分组(groupingBy)、分区(partitioningBy)、汇总(summingInt)等。

3.3 Stream与Optional

Optional是Java 8引入的另一个重要特性,用于避免空指针异常。Stream API在处理过程中经常会与Optional结合使用,以更优雅地处理可能为null的元素。

四、最佳实践

以下是一些使用Java Stream API的具体例子,这些例子将涵盖Stream的基本操作和一些更高级的用法。

例子 1: 筛选和映射

假设我们有一个Person类,包含姓名和年龄,我们想要从一组Person对象中筛选出年龄大于18岁的人,并将他们的名字收集到一个列表中。

/**
 *  筛选和映射
 * @author senfel
 * @date 2024/8/15 17:49 
 * @return void
 */
@Test
public void test(){
    List<Person> people = Arrays.asList(
            new Person("Alice", 30),
            new Person("Bob", 19),
            new Person("Charlie", 17),
            new Person("David", 22)
    );

    List<String> adults = people.stream()
            .filter(p -> p.age > 18)
            .map(Person::getName)
            .collect(Collectors.toList());
    // 输出: [Alice, Bob, David]
    System.out.println(adults); 
}


@Data
class Person {
    String name;
    int age;

    Person(String name, int age) {
        this.name = name;
        this.age = age;
    }
}

例子 2: 排序和收集

现在,如果我们想要根据年龄对这些人进行排序,并将排序后的结果收集到一个新的列表中。

/**
 * 排序和收集
 * @author senfel
 * @date 2024/8/15 17:52 
 * @return void
 */
@Test
public void test2(){
    List<Person> people = Arrays.asList(
            new Person("Alice", 30),
            new Person("Bob", 19),
            new Person("Charlie", 17),
            new Person("David", 22)
    );

    List<Person> sortedPeople = people.stream()
            .sorted(Comparator.comparingInt(Person::getAge))
            .collect(Collectors.toList());
    for (Person p : sortedPeople) {
        //Charlie: 17
        //Bob: 19
        //David: 22
        //Alice: 30
        System.out.println(p.name + ": " + p.age);
    }
}
@Data
class Person {
    String name;
    int age;

    Person(String name, int age) {
        this.name = name;
        this.age = age;
    }
}

例子 3: 分组和汇总

假设我们想要根据年龄将人们分组,并计算每个年龄组中的人数。

/**
 * 分组和汇总
 * @author senfel
 * @date 2024/8/15 17:53 
 * @return void
 */
@Test
public void test3(){
    List<Person> people = Arrays.asList(
            new Person("Alice", 30),
            new Person("Bob", 19),
            new Person("Charlie", 17),
            new Person("David", 22)
    );

    Map<Integer, Long> ageCount = people.stream()
            .collect(Collectors.groupingBy(Person::getAge, Collectors.counting()));

    //Age 17: 1
    //Age 19: 1
    //Age 22: 1
    //Age 30: 1
    ageCount.forEach((age, count) -> System.out.println("Age " + age + ": " + count));
}


@Data
class Person {
    String name;
    int age;

    Person(String name, int age) {
        this.name = name;
        this.age = age;
    }
}

例子 4: 并行流

现在,如果我们想要利用多核处理器来并行处理数据,以提高性能(特别是在处理大量数据时)。

/**
 * 并行流
 * @author senfel
 * @date 2024/8/15 17:54
 * @return void
 */
@Test
public void test4(){
    List<Person> people = Arrays.asList(
            new Person("Alice", 30),
            new Person("Bob", 19),
            new Person("Charlie", 17),
            new Person("David", 22)
    );
    long count = people.parallelStream()
            .filter(p -> p.age > 18)
            .count();
    //Number of adults: 3
    System.out.println("Number of adults: " + count);
}


@Data
class Person {
    String name;
    int age;

    Person(String name, int age) {
        this.name = name;
        this.age = age;
    }
}

注意:并行流的使用需要谨慎,因为它可能不总是比串行流更快,特别是在处理小数据集或数据集已经被很好地分区时。此外,并行流中的操作可能不是线程安全的,因此确保流中的操作是线程安全的非常重要。

五、总结

Java Stream API以其简洁的语法和强大的功能,为Java集合处理带来了全新的体验。通过Stream API,我们可以以声明式的方式处理数据集合,使代码更加简洁、易于理解和维护。同时,结合Lambda表达式和函数式接口,Stream API还能轻松实现复杂的数据转换和聚合逻辑。然而,在使用Stream API时,我们也需要注意其生命周期、并行与串行的选择以及与其他Java特性的结合使用,以充分发挥其优势。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Linux修行路】进程控制——程序替换

目录 ⛳️推荐 一、单进程版程序替换看现象 二、程序替换的基本原理 三、程序替换接口学习 3.1 替换自己写的可执行程序 3.2 第三个参数 envp 验证 ⛳️推荐 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下…

Dify 开源大语言模型(LLM) 应用开发平台如何使用Docker部署与远程访问

目录 ⛳️推荐 前言 1. Docker部署Dify 2. 本地访问Dify 3. Ubuntu安装Cpolar 4. 配置公网地址 5. 远程访问 6. 固定Cpolar公网地址 7. 固定地址访问 ⛳️推荐 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享…

Vue3列表(List)

效果如下图&#xff1a;在线预览 APIs List 参数说明类型默认值bordered是否展示边框booleanfalsevertical是否使用竖直样式booleanfalsesplit是否展示分割线booleantruesize列表尺寸‘small’ | ‘middle’ | ‘large’‘middle’loading是否加载中booleanfalsehoverable是否…

mysql写个分区表

因为表量已经达到1个亿了。现在想做个优化&#xff0c;先按照 create_time 时间进行分区吧。 create_time 是varchar类型。 CREATE TABLE orders (id varchar(40) NOT NULL ,order_no VARCHAR(20) NOT NULL,create_time VARCHAR(20) NOT NULL,amount DECIMAL(10,2) NOT NULL,…

Unity如何使用Spine动画导出的动画

Unity如何使用Spine动画导出的动画 介绍使用版本Spine导出源文件修改Spine3.8.75版本导入Unity的3.8版本Spine的报错Unity辅助修改Json中版本号方式总结 介绍 最近公司在做抖音小程序的小游戏&#xff0c;我们这边动画部分使用的是spine动画&#xff0c;所以会有spine导入的问…

IDEA使用LiveTemplate快速生成方法注释

本文目标&#xff1a;开发人员&#xff0c;在了解利用Live Template动态获取方法输入输出参数、创建日期时间方法的条件下&#xff0c;进行自动生成方法注释&#xff0c;达到自动添加方法注释的程度&#xff1b; 文章目录 1 场景2 要点2.1 新增LiveTemplate模版2.2 模版内容填写…

FFMPEG推流器讲解

FFMPEG重要结构体的讲解 FFMPEG中有六个比较重要的结构体&#xff0c;分别是AVFormatContext、AVOutputFormat、 AVStream、AVCodec、AVCodecContext、AVPacket、AVFrame、AVIOContext结构体&#xff0c;这几个结构体是贯穿着整个FFMPEG核心功能。 AVFormatContext 这个结构…

基于web的大学生一体化服务平台的设计与实现

TOC springboot209基于web的大学生一体化服务平台的设计与实现 第1章 绪论 1.1 课题背景 二十一世纪互联网的出现&#xff0c;改变了几千年以来人们的生活&#xff0c;不仅仅是生活物资的丰富&#xff0c;还有精神层次的丰富。在互联网诞生之前&#xff0c;地域位置往往是人…

1、.Net UI框架:WinUI - .Net宣传系列文章

WinUI(Windows UI Library)是微软提供的一个用于构建Windows应用程序的本机UI平台组件。它与Windows应用SDK紧密相关&#xff0c;允许开发者创建适用于Windows 10及更高版本的应用程序&#xff0c;并且可以发布到Microsoft Store。WinUI 3是最新的一代&#xff0c;它提供了与操…

C# 中 Tuple 与 ValueTuples 之间的区别

在 C# 中&#xff0c;元组和值元组都用于在单个变量中存储多个值。但它们在语法、功能和性能方面存在一些关键差异。 一.Tuples(元组) 元组是一种引用类型&#xff0c;长期以来一直是 .NET 的一部分。它们是使用 System.Tuple 类创建的。 例子 using System; class Program…

养猫家庭必备好物?希喂、霍尼韦尔宠物空气净化器真实测评

随着宠物空气净化器的讨论度越来越高&#xff0c;我也被种草了这款产品。对养宠家庭来说&#xff0c;十分需要这样一款转专门针对宠物毛发清理的工具。准备入手前我在网上做了许多功课&#xff0c;经过一番筛选后&#xff0c;最后希喂、霍尼韦尔两个品牌成功晋级决赛。 在对比…

代理IP为什么不能使用免费代理IP地址?

在跨境业务中&#xff0c;营销人员、广告投手经常利用代理IP防止账号关联与封禁&#xff0c;并且在访问网站时可以隐匿真实 IP 地址&#xff0c;定位目标市场。代理服务器充当中间人掩盖真实的数字足迹&#xff0c;这不仅增强了隐私&#xff0c;也会跨境业务提效提供保障。但是…

基于STM32开发的智能语音助手系统

目录 引言环境准备工作 硬件准备软件安装与配置系统设计 系统架构硬件连接代码实现 初始化代码控制代码应用场景 智能家居控制个人语音助理常见问题及解决方案 常见问题解决方案结论 1. 引言 随着人工智能技术的发展&#xff0c;智能语音助手已经逐渐进入了人们的日常生活。…

【轨物推荐】创新有规律,发明有方法

原创 赵敏 发明方法研究 2020年03月11日 10:38 各位业内朋友&#xff0c;大家好&#xff01; 今年的疫情&#xff0c;对所有的企业都会有冲击&#xff0c;给企业的业务开展带来很多困扰。详细很多企业都在出主意、想办法&#xff0c;设法把疫情造成的损失降到最低。即使在平时…

行业原型:智慧制造:注塑云管工厂

行业原型预览链接&#xff1a;&#xff08;请与班主任联系获取原型文档&#xff09; 文件类型&#xff1a;.rp 支持版本&#xff1a;Axrure RP 8 文档名称&#xff1a;智慧制造&#xff1a;注塑云管工厂 文件大小&#xff1a;1.80 MB 目录内容介绍 文档内容介绍 回复 “211…

均数(mean±SD)与RR/OR值可以合并进行Meta分析吗?

经常有小伙伴问&#xff1a;在做危险因素的Meta分析时&#xff0c;遇到一些文献比较的是病例组和对照组某一指标的均值差异&#xff0c;数据以meanSD形式呈现&#xff0c;而另一些文献则是以OR或RR (95%CI)的形式描述该指标与疾病的关联。这两种数据形式可以一起进行Meta分析吗…

每周心赏|用AI真的可以开挂式求职

马上就要到2024年的金九银十了&#xff0c;正是求职的好时机啊&#xff01; 万事开头难&#xff0c;找工作第一步自然是离不开制作简历了&#xff01; 都说简历是面试的敲门砖&#xff0c;所以大家都很重视。 你不是也曾游走在各大网站中搜寻模板、寻找“大能们”的简历来修…

海运如何实时了解货物的物流轨迹?有什么系统可以实现?

物流轨迹在散货集拼业务中扮演着至关重要的角色&#xff0c;它不仅让客户可以实时掌握货物动态&#xff0c;确保了货物的安全无虞与准时送达&#xff0c;还为企业提供了灵活调整运输策略的依据&#xff0c;有益于运输效率与可靠性的双重提升。同时&#xff0c;通过物流轨迹的即…

RC电路里,电容多久可以充满电

时间常数 τR*C&#xff0c;那么电容需要多久能充满电呢&#xff1f; 例如下图仿真&#xff0c;R1KΩ&#xff0c;C1uF&#xff0c;那么τR*C1ms。 2个时间常数2ms的时间&#xff0c;电容电压充到86% 3个时间常数3ms的时间&#xff0c;电容电压充到95% 通常定义95为充满电。…

前端实现视频流播放:封装一个可复用的HlsPlayer组件

简介 在前端开发中&#xff0c;播放视频流是一个常见的需求&#xff0c;尤其是在需要实时监控或直播的场景中。本文将分享如何封装一个基于hls.js库的Vue组件&#xff0c;以便在任何需要的地方快速引用和播放视频流。 环境准备 首先&#xff0c;确保你的项目中已经安装了Vue…