Java 8 的流库:Filter、Map、FlatMap 及 Optional 的概念与用法

简介: 【6月更文挑战第9天】Java 8 引入了许多强大的新特性,其中流库(Stream API)和 Optional 类极大地简化了集合操作和空值处理。本文将深入探讨 filter、map、flatMap 以及 Optional 的概念和用法,并提供示例代码来展示其实际应用。

Java 8 引入了许多强大的新特性,其中流库(Stream API)和 Optional 类极大地简化了集合操作和空值处理。本文将深入探讨 filter、map、flatMap 以及 Optional 的概念和用法,并提供示例代码来展示其实际应用。

流库(Stream API)

流库提供了一种声明性方式来处理集合数据,支持过滤、映射、聚合等多种操作。流操作可以分为中间操作和终端操作。中间操作返回一个新的流,允许多个操作链式调用;终端操作触发流的处理并返回结果。

1. Filter

filter 是一个中间操作,用于从流中筛选出符合条件的元素。它接收一个谓词(Predicate)作为参数,返回一个由满足谓词条件的元素组成的新流。

示例:过滤出所有的偶数

java复制代码

import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;

public class FilterExample {
    public static void main(String[] args) {
        List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
        List<Integer> evenNumbers = numbers.stream()
                                           .filter(n -> n % 2 == 0)
                                           .collect(Collectors.toList());

        System.out.println(evenNumbers); // 输出: [2, 4, 6, 8, 10]
    }
}

2. Map

map 是一个中间操作,用于将流中的每个元素应用一个函数,并将结果收集到一个新的流中。map 操作通常用于对象的转换和数据的提取。

示例:将字符串列表转换为其长度的列表

java复制代码

import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;

public class MapExample {
    public static void main(String[] args) {
        List<String> words = Arrays.asList("Java", "Stream", "API", "Filter", "Map");
        List<Integer> lengths = words.stream()
                                     .map(String::length)
                                     .collect(Collectors.toList());

        System.out.println(lengths); // 输出: [4, 6, 3, 6, 3]
    }
}

3. FlatMap

flatMap 是一个中间操作,与 map 类似,但它的映射函数会将每个元素转换为一个流,并将多个流合并成一个新的流。flatMap 常用于处理嵌套集合或多对多的映射关系。

示例:将单词列表拆分为字母列表

java复制代码

import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;

public class FlatMapExample {
    public static void main(String[] args) {
        List<String> words = Arrays.asList("Java", "Stream", "API");
        List<String> letters = words.stream()
                                    .flatMap(word -> Arrays.stream(word.split("")))
                                    .collect(Collectors.toList());

        System.out.println(letters); // 输出: [J, a, v, a, S, t, r, e, a, m, A, P, I]
    }
}

Optional 类

Optional 类是 Java 8 引入的一个特殊容器类,用于表示可能为空的值,避免显式的空检查和 NullPointerException

1. 创建 Optional

你可以使用 Optional.ofOptional.ofNullableOptional.empty 方法创建一个 Optional 对象。

java复制代码

import java.util.Optional;

public class OptionalExample {
    public static void main(String[] args) {
        Optional<String> nonEmptyOptional = Optional.of("Hello");
        Optional<String> emptyOptional = Optional.empty();
        Optional<String> nullableOptional = Optional.ofNullable(null);

        System.out.println(nonEmptyOptional); // 输出: Optional[Hello]
        System.out.println(emptyOptional); // 输出: Optional.empty
        System.out.println(nullableOptional); // 输出: Optional.empty
    }
}

2. 使用 Optional

Optional 提供了多种方法来处理可能为空的值,包括 isPresentifPresentorElseorElseGetorElseThrow 等。

示例:使用 Optional 处理可能为空的值

java复制代码

import java.util.Optional;

public class OptionalUsageExample {
    public static void main(String[] args) {
        Optional<String> optionalValue = Optional.of("Hello");

        // 检查值是否存在
        if (optionalValue.isPresent()) {
            System.out.println(optionalValue.get()); // 输出: Hello
        }

        // 使用 ifPresent
        optionalValue.ifPresent(value -> System.out.println(value)); // 输出: Hello

        // 使用 orElse
        String defaultValue = optionalValue.orElse("Default Value");
        System.out.println(defaultValue); // 输出: Hello

        // 使用 orElseGet
        String valueFromSupplier = optionalValue.orElseGet(() -> "Value from Supplier");
        System.out.println(valueFromSupplier); // 输出: Hello

        // 使用 orElseThrow
        try {
            String valueOrException = optionalValue.orElseThrow(() -> new IllegalStateException("No value"));
            System.out.println(valueOrException); // 输出: Hello
        } catch (IllegalStateException e) {
            e.printStackTrace();
        }
    }
}

综合示例

下面是一个综合示例,展示了如何结合使用 Stream API 和 Optional 处理复杂的集合操作:

示例:处理嵌套的集合并处理空值

java复制代码

import java.util.Arrays;
import java.util.List;
import java.util.Optional;
import java.util.stream.Collectors;

public class ComprehensiveExample {
    public static void main(String[] args) {
        List<List<String>> nestedList = Arrays.asList(
                Arrays.asList("Java", "Stream"),
                Arrays.asList("Optional", "Filter", null),
                Arrays.asList("FlatMap", "Map")
        );

        List<String> processedList = nestedList.stream()
                .flatMap(List::stream)
                .map(word -> Optional.ofNullable(word).orElse("Empty"))
                .filter(word -> !word.equals("Empty"))
                .collect(Collectors.toList());

        System.out.println(processedList); // 输出: [Java, Stream, Optional, Filter, FlatMap, Map]
    }
}

总结

Java 8 的流库和 Optional 类为我们提供了强大且简洁的工具,用于处理集合数据和空值。通过 filter、map、flatMap 等中间操作,结合 Optional 类的方法,我们可以轻松地进行数据转换、过滤和空值处理,从而提高代码的可读性和安全性。希望本文对你理解和使用这些重要特性有所帮助。

目录
打赏
0
4
4
0
539
分享
相关文章
|
5月前
|
Java交换map的key和value值
通过本文介绍的几种方法,可以在Java中实现Map键值对的交换。每种方法都有其优缺点,具体选择哪种方法应根据实际需求和场景决定。对于简单的键值对交换,可以使用简单遍历法或Java 8的Stream API;对于需要处理值不唯一的情况,可以使用集合存储或Guava的Multimap。希望本文对您理解和实现Java中的Map键值对交换有所帮助。
87 1
|
6月前
|
优雅地使用Java Map,通过掌握其高级特性和技巧,让代码更简洁。
【10月更文挑战第19天】本文介绍了如何优雅地使用Java Map,通过掌握其高级特性和技巧,让代码更简洁。内容包括Map的初始化、使用Stream API处理Map、利用merge方法、使用ComputeIfAbsent和ComputeIfPresent,以及Map的默认方法。这些技巧不仅提高了代码的可读性和维护性,还提升了开发效率。
213 3
|
6月前
|
详细解析HashMap、TreeMap、LinkedHashMap等实现类,帮助您更好地理解和应用Java Map。
【10月更文挑战第19天】深入剖析Java Map:不仅是高效存储键值对的数据结构,更是展现设计艺术的典范。本文从基本概念、设计艺术和使用技巧三个方面,详细解析HashMap、TreeMap、LinkedHashMap等实现类,帮助您更好地理解和应用Java Map。
121 3
在Java的Map家族中,HashMap和TreeMap各具特色
【10月更文挑战第19天】在Java的Map家族中,HashMap和TreeMap各具特色。HashMap基于哈希表实现,提供O(1)时间复杂度的高效操作,适合性能要求高的场景;TreeMap基于红黑树,提供O(log n)时间复杂度的有序操作,适合需要排序和范围查询的场景。两者在不同需求下各有优势,选择时需根据具体应用场景权衡。
60 2
Java Filter Pattern(过滤器模式)
过滤器模式(Filter Pattern)或标准模式(Criteria Pattern)是一种设计模式,这种模式允许开发人员使用不同的标准来过滤一组对象,通过逻辑运算以解耦的方式把它们连接起来。
1156 0
|
2月前
|
【Java并发】【线程池】带你从0-1入门线程池
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是编写高端CRUD应用。2025年我正在沉淀中,博客更新速度加快,期待与你一起成长。 线程池是一种复用线程资源的机制,通过预先创建一定数量的线程并管理其生命周期,避免频繁创建/销毁线程带来的性能开销。它解决了线程创建成本高、资源耗尽风险、响应速度慢和任务执行缺乏管理等问题。
178 60
【Java并发】【线程池】带你从0-1入门线程池
Java网络编程,多线程,IO流综合小项目一一ChatBoxes
**项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
78 23
|
1月前
|
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
当我们创建一个`ThreadPoolExecutor`的时候,你是否会好奇🤔,它到底发生了什么?比如:我传的拒绝策略、线程工厂是啥时候被使用的? 核心线程数是个啥?最大线程数和它又有什么关系?线程池,它是怎么调度,我们传入的线程?...不要着急,小手手点上关注、点赞、收藏。主播马上从源码的角度带你们探索神秘线程池的世界...
103 0
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
Java社招面试题:一个线程运行时发生异常会怎样?
大家好,我是小米。今天分享一个经典的 Java 面试题:线程运行时发生异常,程序会怎样处理?此问题考察 Java 线程和异常处理机制的理解。线程发生异常,默认会导致线程终止,但可以通过 try-catch 捕获并处理,避免影响其他线程。未捕获的异常可通过 Thread.UncaughtExceptionHandler 处理。线程池中的异常会被自动处理,不影响任务执行。希望这篇文章能帮助你深入理解 Java 线程异常处理机制,为面试做好准备。如果你觉得有帮助,欢迎收藏、转发!
151 14