爆!Java高级特性之Stream API详解

news2024/11/25 4:44:37

爆!Java高级特性之Stream API详解

Java 8引入的Stream API可以说是一个革命性的特性,让我们告别了又臭又长的for循环,迎来了函数式编程的春天。今天就让我们来一起深入了解这个让人又爱又恨的Stream API吧!

什么是Stream?

Stream就像一个高级的迭代器,允许我们以声明式方式处理数据集合。它可以让我们用一种类似SQL查询的方式来操作Java对象。Stream API结合了函数式编程的概念,大大简化了集合操作。

简单来说,Stream就是数据流。我们可以imagin它就像一条传送带,在上面放上要处理的元素,然后让它流过一系列的操作。

创建Stream

创建Stream的方式有很多,我们来看几种常见的:

  1. 从Collection创建
List<String> list = Arrays.asList("a", "b", "c");
Stream<String> stream = list.stream();
  1. 从数组创建
String[] arr = {"a", "b", "c"};
Stream<String> stream = Arrays.stream(arr);
  1. 使用Stream.of()
Stream<String> stream = Stream.of("a", "b", "c");
  1. 生成无限流
Stream<Integer> infiniteStream = Stream.iterate(0, n -> n + 2);

看到这里,有些同学可能会说:"这有什么了不起的?我用for循环一样可以啊!"别急,好戏才刚刚开始。

Stream操作

Stream API提供了丰富的中间操作和终端操作,让我们可以方便地对数据进行各种转换和汇总。

中间操作

中间操作会返回一个新的Stream,我们可以将多个中间操作连接起来形成一个查询。常见的中间操作包括:

  1. filter: 过滤元素
Stream<String> filtered = stream.filter(s -> s.startsWith("a"));
  1. map: 转换元素
Stream<String> mapped = stream.map(String::toUpperCase);
  1. flatMap: 将流中的每个元素转换为一个流,然后把所有流连接起来
Stream<String> flatMapped = stream.flatMap(s -> Arrays.stream(s.split("")));
  1. distinct: 去重
Stream<String> distinct = stream.distinct();
  1. sorted: 排序
Stream<String> sorted = stream.sorted();
  1. peek: 对每个元素执行操作并返回一个新的Stream
Stream<String> peeked = stream.peek(System.out::println);

终端操作

终端操作会遍历流以生成一个结果或副作用。在终端操作之后,流就被使用"光"了,无法再被操作。常见的终端操作包括:

  1. forEach: 遍历每个元素
stream.forEach(System.out::println);
  1. count: 返回流中元素的个数
long count = stream.count();
  1. collect: 将流转换为其他形式
List<String> list = stream.collect(Collectors.toList());
  1. reduce: 将流中元素组合起来
Optional<String> reduced = stream.reduce((s1, s2) -> s1 + s2);
  1. anyMatch, allMatch, noneMatch: 匹配操作
boolean anyStartsWithA = stream.anyMatch(s -> s.startsWith("a"));
  1. findFirst, findAny: 查找操作
Optional<String> first = stream.findFirst();

看到这里,有些同学可能会说:"哇,这么多操作,我脑子都晕了!"别担心,让我们来看一个实际的例子,你就会发现Stream API有多香了。

实战案例

假设我们有一个Person类:

class Person {
    String name;
    int age;
    
    // 构造函数、getter和setter省略
}

现在我们有一个List<Person>,我们想要找出所有年龄大于18岁的人的名字,按字母顺序排序,并且只取前3个。用传统的方式,我们可能会这样写:

List<String> result = new ArrayList<>();
for (Person p : persons) {
    if (p.getAge() > 18) {
        result.add(p.getName());
    }
}
Collections.sort(result);
if (result.size() > 3) {
    result = result.subList(0, 3);
}

看起来不算太糟?那让我们来看看用Stream API怎么写:

List<String> result = persons.stream()
    .filter(p -> p.getAge() > 18)
    .map(Person::getName)
    .sorted()
    .limit(3)
    .collect(Collectors.toList());

怎么样?是不是感觉整个世界都清爽了?这就是Stream API的魅力所在!它让我们的代码更加简洁、易读,而且更加声明式。我们告诉程序"我们想要什么",而不是"怎么去做"。

性能考虑

说到这里,可能有些同学会问:“Stream这么好用,是不是意味着我们应该到处使用它?”

嗯…这个问题问得好!虽然Stream API非常强大,但它并不是万能的。在某些情况下,传统的迭代可能会更快。特别是当我们处理的是基本类型(如int, long)时,使用Stream可能会带来装箱和拆箱的开销。

另外,Stream的延迟执行特性也是把双刃剑。它可以帮我们优化操作,避免不必要的计算。但如果使用不当,也可能导致性能问题。比如:

Stream<Integer> stream = Stream.iterate(0, i -> i + 1);
stream.filter(i -> i % 2 == 0)
      .map(i -> i * 2)
      .limit(10)
      .forEach(System.out::println);

这段代码看起来没什么问题,但实际上它的效率并不高。因为iterate生成的是一个无限流,filtermap操作会被反复执行,直到找到10个符合条件的元素。

一个更高效的写法是:

Stream.iterate(0, i -> i + 2)
      .map(i -> i * 2)
      .limit(10)
      .forEach(System.out::println);

这样我们就避免了不必要的过滤操作。

并行流

Stream API的另一个强大特性是可以轻松地实现并行处理。只需要调用parallel()方法,就可以将串行流转换为并行流:

List<String> result = persons.parallelStream()
    .filter(p -> p.getAge() > 18)
    .map(Person::getName)
    .sorted()
    .limit(3)
    .collect(Collectors.toList());

看起来很诱人是不是?但请记住,并行并不总是更快。在数据量较小或者操作较简单的情况下,并行处理的开销可能会超过其带来的收益。所以在使用并行流之前,一定要进行充分的测试和基准比较。

总结

Stream API无疑是Java 8中最重要的特性之一。它为我们提供了一种新的数据处理方式,让我们的代码更加简洁、易读、高效。但就像所有的技术一样,它也不是银弹。我们需要理解它的工作原理,合理地使用它,才能真正发挥它的威力。

最后,送大家一句话:“Stream API很酷,但请记住,过度使用可能会导致代码可读性下降。保持简单,保持清晰,这才是编程的真谛。”

好了,今天的课程就到这里。如果你觉得这篇文章对你有帮助,别忘了点赞收藏哦!下次我们再来探讨其他Java高级特性。码字不易,你的支持就是我创作的动力!

海码面试 小程序

包含最新面试经验分享,面试真题解析,全栈2000+题目库,前后端面试技术手册详解;无论您是校招还是社招面试还是想提升编程能力,都能从容面对~

好了,今天的课程就到这里。如果你觉得这篇文章对你有帮助,别忘了点赞收藏哦!下次我们再来探讨其他Java高级特性。码字不易,你的支持就是我创作的动力!

海码面试 小程序

包含最新面试经验分享,面试真题解析,全栈2000+题目库,前后端面试技术手册详解;无论您是校招还是社招面试还是想提升编程能力,都能从容面对~

[外链图片转存中…(img-TlNCRLSu-1720181315161)]

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

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

相关文章

应用了网络变压器的PC网卡连接转换器后不好连网,有掉线现象,但外接路由器无问题,可能是什么原因?

Hqst盈盛&#xff08;华强盛&#xff09;电子导读&#xff1a;今天分享的是应用了网络变压器的PC网卡连接转换器后不好连网&#xff0c;有掉线现象&#xff0c;但外接路由器无问题&#xff0c;可能是什么原因呢&#xff1f;如何解决呢&#xff1f; 首先&#xff0c;我们要了解传…

无人机常见故障及维修方法详解

一、无人机故障识别与处理原则 无人机故障识别是维修的第一步&#xff0c;要求操作人员具备基本的无人机系统知识和故障识别能力。在识别故障时&#xff0c;应遵循“先易后难、先外后内、先软件后硬件”的原则。一旦识别出故障&#xff0c;应立即停止飞行&#xff0c;避免进一…

【VUE基础】VUE3第三节—核心语法之computed、watch、watcheffect

computed 接受一个 getter 函数&#xff0c;返回一个只读的响应式 ref 对象。该 ref 通过 .value 暴露 getter 函数的返回值。它也可以接受一个带有 get 和 set 函数的对象来创建一个可写的 ref 对象。 创建一个只读的计算属性 ref&#xff1a; <template><div cl…

Java项目:基于SSM框架实现的校园快递代取管理系统【ssm+B/S架构+源码+数据库+毕业论文】

一、项目简介 本项目是一套基于SSM框架实现的校园快递代取管理系统 包含&#xff1a;项目源码、数据库脚本等&#xff0c;该项目附带全部源码可作为毕设使用。 项目都经过严格调试&#xff0c;eclipse或者idea 确保可以运行&#xff01; 该系统功能完善、界面美观、操作简单、…

树莓派+舵机+pca9685+usb摄像头制作二维云台,图像追踪

使用树莓派+舵机+pca9685+usb摄像头制作二维云台,图像追踪 为什么使用pca9685驱动舵机,而不使用树莓派自带的引脚驱动舵机呢? 因为树莓派无法产生稳定的pwm波,容易造成舵机的抖动 我使用的是树莓派+ubuntu系统+pca9685 1.首先在ubuntu系统中安装相关依赖 sudo apt insta…

海尔智家:科技优秀是一种习惯

海尔智家&#xff1a;科技优秀是一种习惯 2024-06-28 15:19代锡海 6月24日&#xff0c;2023年度国家科学技术奖正式揭晓。海尔智家“温湿氧磁多维精准控制家用保鲜电器技术创新与产业化”项目荣获国家科学技术进步奖&#xff0c;成为家电行业唯一牵头获奖企业。 很多人说&…

odoo 物联网 设备数据采集方案

图一 架构手稿(许老师专属) 图二 架构简图 部署 方案一&#xff1a; odoo业务数据库与设备采集数据库使用一个instance。 缺点&#xff1a;重启pg服务相互影响。 方案二&#xff1a; odoo业务数据库与设备采集数据库独立部署&#xff0c;使用两个instance。 优点&#xff1a;…

VSCode远程服务器

一、安装VSCode Windows安装Visual Studio Code(VS Code)-CSDN博客 二、VSCode中安装Remote-SSH插件 1、在应用商店中搜索Remote - SSH并安装 2、安装后会出现下面标注的图标 三、开始SSH连接 1、点击加号&#xff0c;创建SSH连接 2、输入地址&#xff0c;格式是&#xff1a;…

java集合(1)

目录 一.集合概述 二. 集合体系概述 1. Collection接口 1.1 List接口 1.2 Set接口 2. Map接口 三. ArrayList 1.ArrayList常用方法 2.ArrayList遍历 2.1 for循环 2.2 增强for循环 2.3 迭代器遍历 一.集合概述 我们经常需要存储一些数据类型相同的元素,之前我们学过…

RocketMQ-订阅一致及解决方案

背景 这里借用Rocketmq官方的一句话来描述订阅关系一致: 订阅关系一致指的是同一个消费者分组Group ID下&#xff0c;所有Consumer实例所订阅的Topic和Tag必须完全一致。如果订阅关系不一致&#xff0c;可能导致消息消费逻辑混乱&#xff0c;消息被重复消费或遗漏。 具体的问题…

MySQL中mycat与mha应用

目录 一.Mycat代理服务器 1.Mycat应用场景 2.mycat安装目录结构说明 3.Mycat的常用配置文件 4.Mycat日志 5.mycat 实现读写分离 二.MySQL高可用 1.原理过程 2.MHA软件 3.实现MHA 一.Mycat代理服务器 1.Mycat应用场景 Mycat适用的场景很丰富&#xff0c;以下是几个典型…

【Java11】变量的初始化和内存中的运行机制

成员变量的初始化和内存中的运行机制 系统加载类或创建类的实例时&#xff0c;系统自动为成员变量分配内存空间&#xff0c;然后自动为成员变量指定初始值。 class Person {public String name; // 实例变量public static int eyeNum; // 类变量 }var p1 Person(); var p2 …

【unity实战】Unity中使用A*寻路+有限状态机制作一个俯视角敌人AI

最终效果 文章目录 最终效果前言A*寻路插件介绍下载导入AI插件生成寻路网格节点的类型障碍物寻路测试A*只打印报错信息 代码控制寻路动画配置敌人状态机各种状态脚本效果完结 前言 前面做过有限状态机制作一个敌人AI&#xff1a;【unity实战】在Unity中使用有限状态机制作一个…

【Odoo开源ERP】别把ERP与进销存软件混为一谈

导读&#xff1a;企业使用ERP软件能够实现管理升级&#xff0c;多方信息集成&#xff0c;按照既定策略逻辑运算&#xff0c;生成计划建议&#xff0c;减少人力成本&#xff0c;提高准确率的同时提高经营能力。 ERP&#xff0c;是MRP II的下一代软件&#xff0c;除了MRP II已有的…

WEB安全-靶场

1 需求 2 语法 3 示例 男黑客|在线渗透测试靶场|网络安全培训基地|男黑客安全网 4 参考资料

三万字带你一遍跑通uer

三万字带你一遍跑通uer 参考文档 今天给大家介绍个非常强大的项目uer&#xff0c;集成了许多可以做自然语言的东西&#xff0c;效果的话也非常好&#xff0c;很适合企业级的应用&#xff01; 1. 先将项目uer从github拉取下来&#xff08;zip或git都ok&#xff09; 2. 用pycha…

Vue88-Vuex中的mapActions、mapMutations

一、mapMutations的调用 此时结果不对&#xff0c;因为&#xff1a;若是点击事件不传值&#xff0c;默认传的是event&#xff01;&#xff0c;所以&#xff0c;修改如下&#xff1a; 解决方式1&#xff1a; 解决方式2&#xff1a; 不推荐&#xff0c;写法麻烦&#xff01; 1-…

【代码大全2 选读】看看骨灰级高手消灭 if-else 逻辑的瑞士军刀长啥样

文章目录 1 【写在前面】2 【心法】这把瑞士军刀长啥样3 【示例1】确定某个月份的天数&#xff08;Days-in-Month Example&#xff09;4 【示例2】确定保险费率&#xff08;Insurance Rates Example&#xff09;5 【示例3】灵活的消息格式&#xff08;Flexible-Message-Format …

基于深度学习的图像背景剔除

在过去几年的机器学习领域&#xff0c;我一直想打造真正的机器学习产品。 几个月前&#xff0c;在参加了精彩的 Fast.AI 深度学习课程后&#xff0c;似乎一切皆有可能&#xff0c;我有机会&#xff1a;深度学习技术的进步使许多以前不可能实现的事情成为可能&#xff0c;而且开…

链篦机回转窑球团生产工艺

生球在回转窑氧化焙烧&#xff0c;回转窑头部设有燃烧器&#xff0c;燃料可以采用气体、固体、液体。 来自环冷机一冷却段的高温废气作为二次风进入窑内参与燃烧&#xff0c;烧成成品球进入环冷机。 环冷机采用鼓风冷却&#xff0c;热风风箱分为四段&#xff1a; 一段气体引至…