”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > Java Streams:完全初学者的终极指南

Java Streams:完全初学者的终极指南

发布于2024-11-04
浏览:557

所以,你来这里是为了了解Java中的溪流,但不是人们钓鱼或水流的那种溪流。我们正在谈论数据流,这是 Java 8 中引入的一项强大功能,它使数据处理变得更加容易。无论您是新手还是已经尝试过但无法完全破解,都不必担心。我将用简单易懂的语言引导您完成整个旅程。

准备好?让我们深入了解 Java Streams


Java Streams: The Ultimate Guide for Complete Beginners


Java 中的流是什么?

A Stream 是一种按顺序处理数据的方法。想象一下,您有一个项目列表,并且您想对这些项目执行某些操作(过滤、排序、地图等)。 Stream 可以让您以干净、高效的方式完成这一切。它就像一条装配线,您的数据经过不同的步骤直到得到处理。

关于流要记住的关键事项:

  1. 流不会修改原始数据。将它们视为数据的视图或管道。
  2. 流延迟处理数据,这意味着在您告诉它们产生最终结果之前它们不会做任何实际工作。这避免了不必要的计算。
  3. 流是一次性使用的。一旦流被消耗,它就消失了。如果您想重复使用它,则需要创建一个新的。

为什么使用流?

为什么不直接使用 for 循环或直接操作集合呢?嗯,主要有以下三个原因:

  • 更简洁的代码:无需编写重复的、庞大的循环。流为您提供了一种干净、可读的数据处理方式。
  • 更好的性能:通过惰性求值,流可以更有效地处理数据。它们仅在需要时处理数据,这可以节省处理时间。
  • 函数式风格:流为 Java 带来了一种更具声明性的函数式编程风格,这意味着您专注于您想要做什么,而不是如何

流如何工作?基础知识

我们来看看流操作的两种主要类型:中间终端

1. 中间操作

这些操作准备数据,但不会立即产生最终结果。将这些视为“研讨会”步骤。

  • 筛选()

    这就像一个筛子。它根据条件挑选出元素。例如,如果您只想要整数列表中的偶数,则可以使用 filter().

    java
    Copy code
    List numbers = Arrays.asList(1, 2, 3, 4, 5);
    List evenNumbers = numbers.stream()
                                       .filter(n -> n % 2 == 0)
                                       .collect(Collectors.toList());
    // Output: [2, 4]
    
    

    为什么要过滤? 如果没有filter(),您需要手动循环列表并仅将匹配的元素添加到新列表中。 filter() 让您只需一个干净的步骤即可完成此操作。

  • 地图()

    这是一个变压器。它接受一个元素并返回不同的东西。例如,如果您有一个字符串列表并且想要每个字符串的长度:

    java
    Copy code
    List words = Arrays.asList("apple", "banana", "cherry");
    List lengths = words.stream()
                                 .map(String::length)
                                 .collect(Collectors.toList());
    // Output: [5, 6, 6]
    
    

    为什么要使用map? 当您需要将每个元素转换为其他内容时使用map(),例如将字符串列表转换为其长度的列表。

  • 清楚的()

    这就像一个重复的过滤器。这会从流中删除重复的元素。

    java
    Copy code
    List numbers = Arrays.asList(1, 2, 2, 3, 4, 4, 5);
    List distinctNumbers = numbers.stream()
                                           .distinct()
                                           .collect(Collectors.toList());
    // Output: [1, 2, 3, 4, 5]
    
    

    为什么不同? 在普通列表中,您需要手动检查重复项。 unique() 在一行中为您完成此操作。

  • 排序()

    这会按自然顺序(或自定义顺序,如果您愿意)对数据进行排序。

    java
    Copy code
    List names = Arrays.asList("Charlie", "Alice", "Bob");
    List sortedNames = names.stream()
                                    .sorted()
                                    .collect(Collectors.toList());
    // Output: ["Alice", "Bob", "Charlie"]
    
    

    为什么要排序? 不用自己编写排序逻辑,sorted() 会为您处理。

2. 终端操作

这些是产生最终结果的,它们触发整个流的处理。将这些视为“退出点”。

  • 收集()

    这是最常见的终端操作。它收集流的结果并将它们放入列表、集合或其他集合中。

    java
    Copy code
    List names = Arrays.asList("Charlie", "Alice", "Bob");
    List upperNames = names.stream()
                                   .map(String::toUpperCase)
                                   .collect(Collectors.toList());
    // Output: ["CHARLIE", "ALICE", "BOB"]
    
    

    为什么要收集? 您几乎总是使用collect() 将流的结果收集到集合中。这是你的最后一站。

  • forEach()

    如果您不需要结果而只想对每个项目执行操作(例如打印它们),那么 forEach() 是您的朋友。

    java
    Copy code
    numbers.stream()
           .forEach(System.out::println);
    
    

    为什么使用 forEach? 这非常适合副作用,例如将数据打印到控制台或写入文件。

  • 减少()

    reduce() 获取一堆数据并将其归结为单个结果。例如,对数字列表求和:

    java
    Copy code
    int sum = numbers.stream()
                     .reduce(0, Integer::sum);
    // Output: 15
    
    

    为什么要减少? 当您需要将值组合或累积为单个结果时,reduce() 是您的首选。


其他类型的流

并非所有流都是从集合创建的。 Java提供了其他类型的流来处理各种类型的数据:


Java Streams: The Ultimate Guide for Complete Beginners


  1. IntStream、LongStream、DoubleStream

    这些流专门用于处理原始类型。您可以使用这些值来避免性能损失,而不是像 Stream 这样对值进行装箱和拆箱。

    例子:

    java
    Copy code
    IntStream intStream = IntStream.of(1, 2, 3, 4);
    int sum = intStream.sum();  // Output: 10
    
    
  2. 文件流

    您可以使用 Files.lines().
    从文件创建流

    java
    Copy code
    try (Stream lines = Files.lines(Paths.get("data.txt"))) {
        lines.forEach(System.out::println);
    } catch (IOException e) {
        e.printStackTrace();
    }
    
    

    为什么使用文件流?处理大文件时,将所有数据加载到内存中可能效率不高。使用流可以让您逐行处理它。


何时使用流?

  • 转换数据:当你需要修改集合的每个元素时。
  • 过滤:当您只想选择符合特定条件的数据时。
  • 聚合数据:当您需要将集合缩减为单个结果(例如求和、平均值)时。
  • 并行处理:流也支持并行性。使用 .parallelStream(),您可以将任务拆分到多个线程以加快处理速度。

流与循环:为什么不直接使用循环?

好问题!我们来比较一下:

  1. 可读性:使用 Streams,您可以专注于您想要做什么,而不是如何。循环往往会让您编写大量额外的样板代码(例如计数器和条件)。
  2. 性能:流经过优化,可以有效地处理大数据,特别是在延迟计算和并行性方面。循环不提供这种开箱即用的优化。
  3. 灵活性:流允许您以干净、实用的方式链接操作(如过滤、映射和缩减)。循环需要您在其中嵌套更多逻辑。

总结

Java 中的流就是为了简化处理数据的方式。它们使您的代码在使用集合时更具可读性、更易于维护且更高效。无论您是过滤、转换还是减少数据,Streams 都能为您提供清晰、简单的方法,从而消除繁琐的循环和手动工作的需要。

既然您已经掌握了 Streams 的基础知识,为什么还要停在这里呢?在 Twitter、LinkedIn 上关注我,或者查看我的博客以获取更多 Java 技巧,这些技巧将使您立即成为专业人士!如果您发现本指南有帮助,请与其他开发人员分享 - 因为分享就是关爱!


准备好尝试了吗?让我们在您的下一个项目中让该 Stream 流动起来!

版本声明 本文转载于:https://dev.to/wittedtech-by-harshit/java-streams-the-ultimate-guide-for-complete-beginners-40e2?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 如何在 MySQL SELECT 查询中使用 IF 语句:语法和最佳实践指南
    如何在 MySQL SELECT 查询中使用 IF 语句:语法和最佳实践指南
    MySQL SELECT 查询中的 IF 语句用法尝试在 MySQL SELECT 查询中实现 IF 语句时,您可能会遇到困难,因为具体语法要求。在 MySQL 中,传统的 IF/THEN/ELSE 结构仅在存储过程和函数中可用。要解决此问题,重构查询至关重要。查询中支持的 IF() 函数主要用于根...
    编程 发布于2024-11-17
  • 如何在Python中生成特定范围内的唯一随机数?
    如何在Python中生成特定范围内的唯一随机数?
    生成范围内的唯一随机数生成随机数时,确保每个数字都是唯一的可能是一个挑战。虽然可以使用条件语句来检查重复项,但在处理大范围或大数字时,这种方法会变得很麻烦。生成唯一随机数列表的一种简单方法是使用 Python 的 random.sample( ) 功能。该函数有两个参数:总体(生成随机数的数字范围)...
    编程 发布于2024-11-17
  • 如何使用占位符和参数执行带有“WHERE...IN”的 PDO 查询?
    如何使用占位符和参数执行带有“WHERE...IN”的 PDO 查询?
    PDO 查询“WHERE...IN”为了使用 PDO 增强数据库访问,许多开发人员遇到了挑战,特别是“WHERE...IN” IN”查询。让我们深入研究其中的复杂性,并发现在 PDO 准备好的语句中使用项目列表的正确方法。“WHERE... IN”难题考虑一个场景,您需要根据表单中已检查项目的列表从...
    编程 发布于2024-11-17
  • Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta:列偏移的删除和恢复Bootstrap 4 在其 Beta 1 版本中引入了重大更改柱子偏移了。然而,随着 Beta 2 的后续发布,这些变化已经逆转。从 offset-md-* 到 ml-auto在 Bootstrap 4 Beta 1 中, offset-md-*...
    编程 发布于2024-11-17
  • 如何高效地访问和删除Python字典中的任意元素?
    如何高效地访问和删除Python字典中的任意元素?
    访问 Python 字典中的任意元素在 Python 中,字典将键值对存储在无序集合中。如果字典不为空,则可以使用以下语法访问任意(随机)元素:mydict[list(mydict.keys())[0]]但是,这这种方法可能冗长且低效,特别是当您需要执行多次迭代时。让我们探索更有效的方法来实现此目的...
    编程 发布于2024-11-17
  • 如何使用 jQuery 制作背景颜色动画?
    如何使用 jQuery 制作背景颜色动画?
    使用 jQuery 淡化背景颜色引人注目的网站元素通常需要微妙的动画,例如淡入和淡出。虽然 jQuery 广泛用于动画文本内容,但它也可用于动态增强背景颜色。在 jQuery 中淡入/淡出背景颜色进行操作要使用 jQuery 设置元素的背景颜色,您首先需要合并 jQueryUI 库。集成后,可以使用...
    编程 发布于2024-11-17
  • 如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    MySQL配置不正确:相对路径的问题在Django中运行python manage.py runserver时,可能会遇到以下错误:ImproperlyConfigured: Error loading MySQLdb module: dlopen(/Library/Python/2.7/site-...
    编程 发布于2024-11-17
  • 如何在 PHP 中组合两个关联数组,同时保留唯一 ID 并处理重复名称?
    如何在 PHP 中组合两个关联数组,同时保留唯一 ID 并处理重复名称?
    在 PHP 中组合关联数组在 PHP 中,将两个关联数组组合成一个数组是一项常见任务。考虑以下请求:问题描述:提供的代码定义了两个关联数组,$array1 和 $array2。目标是创建一个新数组 $array3,它合并两个数组中的所有键值对。 此外,提供的数组具有唯一的 ID,而名称可能重合。要求...
    编程 发布于2024-11-17
  • 除了“if”语句之外:还有哪些地方可以在不进行强制转换的情况下使用具有显式“bool”转换的类型?
    除了“if”语句之外:还有哪些地方可以在不进行强制转换的情况下使用具有显式“bool”转换的类型?
    无需强制转换即可上下文转换为 bool您的类定义了对 bool 的显式转换,使您能够在条件语句中直接使用其实例“t”。然而,这种显式转换提出了一个问题:“t”在哪里可以在不进行强制转换的情况下用作 bool?上下文转换场景C 标准指定了四种值可以根据上下文转换为的主要场景bool:语句:if、whi...
    编程 发布于2024-11-17
  • 在 Go 中嵌入结构:指针还是值?何时使用哪个?
    在 Go 中嵌入结构:指针还是值?何时使用哪个?
    Go 中嵌入结构:何时使用指针当考虑将一个结构嵌入另一个结构时,决定是否使用指针或者出现嵌入字段的值。本文探讨了这种实现选择的细微差别,并提供示例来说明潜在的好处和影响。通过指针嵌入Go 规范允许将结构体嵌入为指针或价值观。对于非接口类型,允许将匿名字段指定为类型名称 T 或指向非接口类型名称 *T...
    编程 发布于2024-11-17
  • PHP 中可以不继承地修改类方法吗?
    PHP 中可以不继承地修改类方法吗?
    我可以在没有继承的情况下对类进行猴子修补吗?您可能会遇到需要修改类或其方法而没有典型继承选项的情况。以下面的类为例:class third_party_library { function buggy_function() { return 'bad result'; ...
    编程 发布于2024-11-17
  • 如何使用纯 JavaScript 自动调整文本区域大小?
    如何使用纯 JavaScript 自动调整文本区域大小?
    Textarea 自动高度这个问题旨在消除文本区域的滚动条并调整其高度以匹配其中的内容。提供了使用纯JavaScript代码的解决方案:function auto_grow(element) { element.style.height = "5px"; element....
    编程 发布于2024-11-17
  • 什么时候手动调用析构函数是合法的做法?
    什么时候手动调用析构函数是合法的做法?
    什么时候手动调用析构函数是合理的?手动调用析构函数表明存在缺陷的设计这一概念经常被断言。然而,这引出了一个问题:这条规则有例外吗?反例:需要手动析构函数调用的情况确实,在某些情况下有必要这样做显式调用析构函数:1.受控内存释放:当内存分配和释放独立于对象构造和销毁进行管理时,手动析构函数调用变得至关...
    编程 发布于2024-11-17
  • 大批
    大批
    方法是可以在对象上调用的 fns 数组是对象,因此它们在 JS 中也有方法。 slice(begin):将数组的一部分提取到新数组中,而不改变原始数组。 let arr = ['a','b','c','d','e']; // Usecase: Extract till index p...
    编程 发布于2024-11-17
  • 使用 Go Module 部署 Go Cloud Functions 时如何解决“Build failed: go: parsing /models/go.mod: open /models/go.mod: no such file or directory\”错误?
    使用 Go Module 部署 Go Cloud Functions 时如何解决“Build failed: go: parsing /models/go.mod: open /models/go.mod: no such file or directory\”错误?
    使用 Go 模块部署 Google Cloud Function 错误尝试使用 Go 模块在 Go 1.11 中部署 Google Cloud Function 时,开发者可能会遇到错误,“构建失败:go:解析/models/go.mod:打开/models/go.mod:没有这样的文件或目录。&q...
    编程 发布于2024-11-17

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3