当前位置: 首页 > news >正文

Java Stream API详解:高效处理集合数据的利器

引言

Java 8引入了许多新特性,其中最为显著的莫过于Lambda表达式和Stream API。Stream API提供了一种高效、简洁的方法来处理集合数据,使代码更加简洁明了,且具有较高的可读性和可维护性。本文将深入探讨Java Stream API的使用,包括基础概念、常用操作、并行处理、实战案例以及最佳实践等内容。

目录

  1. 什么是Stream API
  2. Stream API的基础操作
    • 创建Stream
    • 中间操作
    • 终端操作
  3. Stream API的高级操作
    • 排序
    • 筛选
    • 映射
    • 规约
    • 收集
  4. 并行Stream
  5. Stream API实战案例
    • 处理集合数据
    • 文件操作
    • 数据库操作
  6. Stream API的最佳实践
  7. 常见问题与解决方案
  8. 总结

什么是Stream API

Stream API是Java 8引入的一种用于处理集合数据的抽象,它允许以声明性方式(类似SQL语句)来处理数据。Stream API提供了许多强大的操作,可以用来对集合进行过滤、排序、映射、规约等操作,极大地简化了代码。

特点

  • 声明性编程:使用Stream API可以以声明性的方式编写代码,减少样板代码。
  • 链式调用:Stream API的操作可以链式调用,提高代码的可读性。
  • 惰性求值:中间操作是惰性求值的,只有在执行终端操作时才会进行计算。
  • 并行处理:支持并行处理,可以充分利用多核CPU的优势。

Stream API的基础操作

创建Stream

Stream API提供了多种方式来创建Stream,常见的有以下几种:

  1. 从集合创建
List<String> list = Arrays.asList("a", "b", "c");
Stream<String> stream = list.stream();
  1. 从数组创建
String[] array = {"a", "b", "c"};
Stream<String> stream = Arrays.stream(array);
  1. 使用Stream.of
Stream<String> stream = Stream.of("a", "b", "c");
  1. 使用Stream.generate
Stream<Double> stream = Stream.generate(Math::random).limit(10);
  1. 使用Stream.iterate
Stream<Integer> stream = Stream.iterate(0, n -> n + 2).limit(10);

中间操作

中间操作用于转换Stream,是惰性求值的,常见的中间操作有以下几种:

  1. filter:用于过滤元素。
Stream<String> stream = list.stream().filter(s -> s.startsWith("a"));
  1. map:用于映射每个元素到对应的结果。
Stream<String> stream = list.stream().map(String::toUpperCase);
  1. flatMap:用于将每个元素转换为Stream,然后合并成一个Stream。
Stream<String> stream = list.stream().flatMap(s -> Stream.of(s.split("")));
  1. distinct:用于去重。
Stream<String> stream = list.stream().distinct();
  1. sorted:用于排序。
Stream<String> stream = list.stream().sorted();
  1. peek:用于在处理过程中查看每个元素。
Stream<String> stream = list.stream().peek(System.out::println);

终端操作

终端操作用于启动Stream的计算,并生成结果,常见的终端操作有以下几种:

  1. forEach:对每个元素执行操作。
list.stream().forEach(System.out::println);
  1. collect:将Stream转换为其他形式。
List<String> result = list.stream().collect(Collectors.toList());
  1. reduce:将Stream中的元素规约成一个值。
Optional<String> result = list.stream().reduce((s1, s2) -> s1 + s2);
  1. toArray:将Stream转换为数组。
String[] array = list.stream().toArray(String[]::new);
  1. count:计算元素个数。
long count = list.stream().count();
  1. anyMatchallMatchnoneMatch:用于匹配判断。
boolean anyMatch = list.stream().anyMatch(s -> s.startsWith("a"));
boolean allMatch = list.stream().allMatch(s -> s.startsWith("a"));
boolean noneMatch = list.stream().noneMatch(s -> s.startsWith("a"));
  1. findFirstfindAny:用于查找元素。
Optional<String> first = list.stream().findFirst();
Optional<String> any = list.stream().findAny();

Stream API的高级操作

排序

使用sorted方法对Stream进行排序,可以传入一个比较器。

List<String> list = Arrays.asList("b", "c", "a");
List<String> sortedList = list.stream().sorted().collect(Collectors.toList());
// 逆序排序
List<String> sortedListDesc = list.stream().sorted(Comparator.reverseOrder()).collect(Collectors.toList());

筛选

使用filter方法对Stream中的元素进行筛选。

List<String> list = Arrays.asList("a", "b", "c");
List<String> filteredList = list.stream().filter(s -> s.startsWith("a")).collect(Collectors.toList());

映射

使用map方法对Stream中的元素进行映射。

List<String> list = Arrays.asList("a", "b", "c");
List<String> mappedList = list.stream().map(String::toUpperCase).collect(Collectors.toList());

规约

使用reduce方法对Stream中的元素进行规约。

List<String> list = Arrays.asList("a", "b", "c");
String result = list.stream().reduce("", (s1, s2) -> s1 + s2);

收集

使用collect方法将Stream转换为其他形式。

List<String> list = Arrays.asList("a", "b", "c");
List<String> collectedList = list.stream().collect(Collectors.toList());
Set<String> collectedSet = list.stream().collect(Collectors.toSet());
String joinedString = list.stream().collect(Collectors.joining(","));

并行Stream

并行Stream可以充分利用多核CPU的优势,提高数据处理的效率。可以使用parallelStream方法创建并行Stream。

List<String> list = Arrays.asList("a", "b", "c");
List<String> parallelList = list.parallelStream().map(String::toUpperCase).collect(Collectors.toList());

也可以使用parallel方法将普通Stream转换为并行Stream。

List<String> list = Arrays.asList("a", "b", "c");
List<String> parallelList = list.stream().parallel().map(String::toUpperCase).collect(Collectors.toList());

需要注意的是,并行Stream并不是总是比串行Stream更快,具体需要根据具体情况进行测试。

Stream API实战案例

处理集合数据

案例一:过滤并转换集合

给定一个包含若干字符串的集合,过滤掉长度小于3的字符串,并将剩余字符串转换为大写。

List<String> list = Arrays.asList("a", "ab", "abc", "abcd");
List<String> result = list.stream().filter(s -> s.length() >= 3).map(String::toUpperCase).collect(Collectors.toList());
System.out.println(result); // 输出:[ABC, ABCD]
案例二:计算平均值

给定一个包含若干整数的集合,计算所有整数的平均值。

List<Integer> list = Arrays.asList(1, 2, 3, 4, 5);
OptionalDouble average = list.stream().mapToInt(Integer::intValue).average();
average.ifPresent(System.out::println); // 输出:3.0

文件操作

案例三:读取文件内容

使用Stream API

读取文件内容并输出到控制台。

try (Stream<String> lines = Files.lines(Paths.get("example.txt"))) {lines.forEach(System.out::println);
} catch (IOException e) {e.printStackTrace();
}
案例四:统计单词出现次数

读取文件内容并统计每个单词出现的次数。

try (Stream<String> lines = Files.lines(Paths.get("example.txt"))) {Map<String, Long> wordCount = lines.flatMap(line -> Arrays.stream(line.split("\\W+"))).collect(Collectors.groupingBy(String::toLowerCase, Collectors.counting()));wordCount.forEach((word, count) -> System.out.println(word + ": " + count));
} catch (IOException e) {e.printStackTrace();
}

数据库操作

案例五:处理数据库查询结果

假设我们有一个数据库表users,包含字段idnameage。我们可以使用Stream API处理查询结果。

List<User> users = queryDatabase();
List<String> names = users.stream().filter(user -> user.getAge() > 18).map(User::getName).collect(Collectors.toList());
System.out.println(names);

Stream API的最佳实践

  1. 避免不必要的并行化:并行Stream并不是总是更快,应该根据具体情况进行选择。
  2. 合理使用中间操作和终端操作:中间操作是惰性求值的,只有在执行终端操作时才会进行计算。
  3. 注意Stream的可复用性:Stream一旦被消费就不能再使用,如果需要复用,可以考虑将Stream转换为集合再使用。
  4. 使用合适的收集器Collectors类提供了多种收集器,可以根据具体需求选择合适的收集器。
  5. 处理异常:在使用Stream API时,需要处理可能出现的异常,尤其是在文件操作和数据库操作中。

常见问题与解决方案

Stream已关闭

Stream一旦被消费就不能再使用,如果需要复用,可以考虑将Stream转换为集合再使用。

List<String> list = Arrays.asList("a", "b", "c");
Stream<String> stream = list.stream();
stream.forEach(System.out::println);
stream.forEach(System.out::println); // 会抛出IllegalStateException

性能问题

并行Stream并不是总是比串行Stream更快,具体需要根据具体情况进行测试。可以使用ForkJoinPool来优化并行Stream的性能。

ForkJoinPool customThreadPool = new ForkJoinPool(4);
customThreadPool.submit(() ->list.parallelStream().forEach(System.out::println)
).get();

内存泄漏

在使用Stream API处理大数据量时,需要注意内存泄漏的问题。可以使用close方法关闭Stream,或者使用try-with-resources语句自动关闭Stream。

try (Stream<String> lines = Files.lines(Paths.get("example.txt"))) {lines.forEach(System.out::println);
} catch (IOException e) {e.printStackTrace();
}

总结

本文详细介绍了Java Stream API的使用,包括基础操作、高级操作、并行处理、实战案例以及最佳实践等内容。通过合理利用Stream API,开发者可以大大简化代码,提高代码的可读性和可维护性,同时还可以提高数据处理的效率。希望本文对你在Java开发中的Stream API使用有所帮助。

Java Stream API是处理集合数据的强大工具,通过灵活运用各种操作,可以实现高效的数据处理和流式计算。如果你还没有使用过Stream API,建议尽快学习和掌握这一强大的工具,将其应用到你的项目中,提升开发效率和代码质量。

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 休息时间c++
  • Java对象创建究竟是在栈上还是堆上??
  • FPGA上板项目(一)——点灯熟悉完整开发流程、ILA在线调试
  • 【C语言】《回调函数》详细解析
  • 探索最佳海外代理服务商!你知道哪些?
  • Java面试题系列 - 第10天
  • Python编程实例-Python的隐藏特性
  • 数据建设实践之大数据平台(三)安装hadoop
  • Ubuntu 安装配置与调优 Docker 并支持 IPv6
  • system V共享内存【Linux】
  • 如何看待AI机器人取代人工拨打电话
  • 网络编程:基本概念udp
  • 嵌入式开发过程中,常见报错以及解决方法
  • 华为OD机试D卷 --跳格子3--24年OD统一考试(Java JS Python C C++)
  • ArduPilot开源飞控之AP_Mount_Topotek
  • CSS进阶篇--用CSS开启硬件加速来提高网站性能
  • ES6 学习笔记(一)let,const和解构赋值
  • FastReport在线报表设计器工作原理
  • Github访问慢解决办法
  • Hibernate【inverse和cascade属性】知识要点
  • Js基础知识(一) - 变量
  • JS实现简单的MVC模式开发小游戏
  • laravel with 查询列表限制条数
  • python 装饰器(一)
  • Python代码面试必读 - Data Structures and Algorithms in Python
  • Spark学习笔记之相关记录
  • tab.js分享及浏览器兼容性问题汇总
  • WinRAR存在严重的安全漏洞影响5亿用户
  • 从地狱到天堂,Node 回调向 async/await 转变
  • 机器人定位导航技术 激光SLAM与视觉SLAM谁更胜一筹?
  • 容器服务kubernetes弹性伸缩高级用法
  • 时间复杂度与空间复杂度分析
  • 使用权重正则化较少模型过拟合
  • 在weex里面使用chart图表
  • ​​​​​​​STM32通过SPI硬件读写W25Q64
  • ​zookeeper集群配置与启动
  • ​Z时代时尚SUV新宠:起亚赛图斯值不值得年轻人买?
  • ‌JavaScript 数据类型转换
  • (3)(3.5) 遥测无线电区域条例
  • (3)医疗图像处理:MRI磁共振成像-快速采集--(杨正汉)
  • (6)设计一个TimeMap
  • (C语言)深入理解指针2之野指针与传值与传址与assert断言
  • (二十九)STL map容器(映射)与STL pair容器(值对)
  • (非本人原创)史记·柴静列传(r4笔记第65天)
  • (分享)自己整理的一些简单awk实用语句
  • (附源码)springboot炼糖厂地磅全自动控制系统 毕业设计 341357
  • (回溯) LeetCode 131. 分割回文串
  • (回溯) LeetCode 78. 子集
  • (论文阅读32/100)Flowing convnets for human pose estimation in videos
  • (删)Java线程同步实现一:synchronzied和wait()/notify()
  • (学习日记)2024.03.12:UCOSIII第十四节:时基列表
  • (一)Neo4j下载安装以及初次使用
  • (幽默漫画)有个程序员老公,是怎样的体验?
  • (转)甲方乙方——赵民谈找工作
  • ./configure,make,make install的作用