Java 8 到 Java 21 系列之 Optional 类型:优雅地处理空值(Java 8)

news2025/5/10 15:35:59

Java 8 到 Java 21 系列之 Optional 类型:优雅地处理空值(Java 8)

系列目录

  • Java8 到 Java21 系列之 Lambda 表达式:函数式编程的开端(Java 8)
  • Java 8 到 Java 21 系列之 Stream API:数据处理的新方式(Java 8)
  • Java 8 到 Java 21 系列之 Optional 类型:优雅地处理空值(Java 8)
  • Java 8 到 Java 21 系列之 新日期时间 API:精确的时间管理(Java 8) 更新中
  • Java 8 到 Java 21 系列之 模块化系统:构建模块化的 Java 应用(Java 9) 更新中
  • Java 8 到 Java 21 系列之 JShell:即时运行 Java 代码(Java 9) 更新中
  • Java 8 到 Java 21 系列之 局部变量类型推断:var 关键字的妙用(Java 10) 更新中
  • Java 8 到 Java 21 系列之 HTTP Client API:现代网络通信的基础(Java 11) 更新中
  • Java 8 到 Java 21 系列之 ZGC:低延迟垃圾收集器的秘密(Java 11) 更新中
  • Java 8 到 Java 21 系列之 Switch 表达式的进化(Java 12) 更新中
  • Java 8 到 Java 21 系列之 文本块:轻松管理多行字符串(Java 13) 更新中
  • Java 8 到 Java 21 系列之 instanceof 模式匹配:简化类型检查(Java 14) 更新中
  • Java 8 到 Java 21 系列之 Records:数据类的全新体验(Java 14) 更新中
  • Java 8 到 Java 21 系列之 密封类:限制继承的艺术(Java 15) 更新中
  • Java 8 到 Java 21 系列之 外部函数与内存 API:无缝集成本地代码(Java 17) 更新中
  • Java 8 到 Java 21 系列之 Sealed Classes 正式登场:增强类型安全性(Java 17) 更新中
  • Java 8 到 Java 21 系列之 强封装 JDK 内部 API:保护你的应用程序(Java 17) 更新中
  • Java 8 到 Java 21 系列之 增强的伪随机数生成器:更高质量的随机数(Java 17) 更新中
  • Java 8 到 Java 21 系列之 虚拟线程:并发编程的新纪元(Java 21) 更新中
  • Java 8 到 Java 21 系列之 分代 ZGC 优化:迈向更高性能(Java 21) 更新中
  • Java 8 到 Java 21 系列之 序列集合 API:简化集合操作(Java 21) 更新中

摘要与引言

在软件开发的实践中,空指针异常(NullPointerException)是开发者常常遇到的问题之一。这类异常往往会导致程序崩溃,尤其是在大型系统中定位问题所在会变得非常棘手。为了帮助开发者更优雅地处理可能为空的值,Java 8 引入了 Optional 类型,这一特性极大地改善了代码的健壮性和可读性。


1 什么是 Optional

Optional<T> 是一个容器类,它可以包含一个非空值或者不包含任何值(即空值)。它的主要目的是提供一种更加安全和清晰的方式来处理可能为空的值,从而减少空指针异常的风险。

1.1 空指针异常演示

下面的例子展示了未使用 Optional 时可能出现的空指针异常。

/**
 * Optional单元测试案例
 *
 * @author JunLiang
 */
@DisplayName("Optional单元测试案例")
public class OptionalTest {
    
    @DisplayName("空指针测试")
    @Test
    public void nullPointerTest() {
        UserService userService = new UserService();
        User user = userService.getUserById(1L);
        System.out.println(user.getName()); // 可能抛出NullPointerException
    }

    class User {
        private String name;

        public User(String name) {
            this.name = name;
        }

        public String getName() {
            return name;
        }
    }

    class UserService {
        public User getUserById(Long id) {
            // 假设这里查询数据库,如果找不到用户则返回null
            return null;
        }
    }

}

直接运行单元测试示例报空指针异常
在这里插入图片描述

1.2 使用 Optional 处理

现在我们将上面的示例修改为使用 Optional 来避免空指针异常。
当查询对应的用户为空时,给用户名默认值guest

/**
 * Optional单元测试案例
 *
 * @author JunLiang
 */
@DisplayName("Optional单元测试案例")
public class OptionalTest {

    @DisplayName("空指针测试 - 使用Optional改进")
    @Test
    public void nullPointerTestWithOptional() {
        UserService userService = new UserService();

        // 如果用户为空,则提供默认值 "Guest"
        Optional<User> optionalUser = Optional.ofNullable(userService.getUserById(1L));
        String userName = optionalUser.map(User::getName).orElse("Guest");

        // 打印用户名或默认值
        System.out.println("默认用户名: " + userName);
    }

   class User {
        private String name;

        public User(String name) {
            this.name = name;
        }

        public String getName() {
            return name;
        }
    }

    class UserService {
        public User getUserById(Long id) {
            // 假设这里查询数据库,如果找不到用户则返回null
            return null; // 为了演示,总是返回null
        }
    }
}

执行结果
在这里插入图片描述
当查询对应的用户为空时,给用配合try捕获异常进行报错提示

/**
 * Optional单元测试案例
 *
 * @author JunLiang
 */
@DisplayName("Optional单元测试案例")
public class OptionalTest {

    @DisplayName("空指针测试 - 使用Optional改进")
    @Test
    public void nullPointerTestWithOptional() {
        UserService userService = new UserService();

        // 如果用户为空,则抛出自定义异常
        try {
            // 使用Optional包装可能为空的返回值
            Optional<User> optionalUser = Optional.ofNullable(userService.getUserById(1L));

            // 尝试获取用户名,如果用户为空则抛出自定义异常
            String userName = optionalUser.map(User::getName)
                .orElseThrow(() -> new RuntimeException("用户不存在"));

            // 打印用户名
            System.out.println("用户名: " + userName);
        } catch (RuntimeException e) {
            // 捕获运行时异常并打印日志
            System.out.println("捕获到运行时异常: " + e.getMessage());
        }
        
    }

    class User {
        private String name;

        public User(String name) {
            this.name = name;
        }

        public String getName() {
            return name;
        }
    }

    class UserService {
        public User getUserById(Long id) {
            // 假设这里查询数据库,如果找不到用户则返回null
            return null; // 为了演示,总是返回null
        }
    }
}

执行结果
在这里插入图片描述

通过上述介绍可以看出,Optional 提供了一种优雅的方式处理潜在的空值。在实际项目中,我们可以在方法返回值、数据传输对象(DTO)、以及链式调用等场景下充分利用 Optional 来提高代码质量。

例如,在服务层方法返回用户信息时,可以返回 Optional<User> 而不是直接返回 User,这样可以明确表达返回值可能是空的意图,并且鼓励调用方考虑如何处理这种情况。


2 常见Optional使用场景

2.1 创建 Optional 对象

  • 有值的情况:当你确定某个变量不是 null 时,可以使用 Optional.of()
    Optional<String> nonNullOptional = Optional.of("Hello");
    
  • 可能为空的情况:当你不确定某个变量是否为 null 时,应该使用 Optional.ofNullable()
    String nullableValue = null;
    Optional<String> nullableOptional = Optional.ofNullable(nullableValue);
    
  • 无值的情况:当需要表示没有值时,可以使用 Optional.empty()
    Optional<String> emptyOptional = Optional.empty();
    

2.2 检查是否存在值

  • 使用 isPresent() 或者 Java 11 引入的 isEmpty() 方法来检查 Optional 是否包含值。
    if (nonNullOptional.isPresent()) {
        System.out.println("Value is present: " + nonNullOptional.get());
    }
    

2.3 执行动作如果存在值

  • 使用 ifPresent() 方法在 Optional 包含值时执行一个操作。
    nonNullOptional.ifPresent(value -> System.out.println("Value is present: " + value));
    

2.4 获取值或默认值

  • 使用 orElse() 提供一个默认值,在 Optional 为空时返回该默认值。
    String result = nullableOptional.orElse("Default Value");
    
  • 使用 orElseGet() 提供一个 Supplier 来延迟计算默认值。
    String computedResult = nullableOptional.orElseGet(() -> computeDefaultValue());
    

2.5. 抛出异常

  • 使用 orElseThrow()Optional 为空时抛出异常。
    String value = nullableOptional.orElseThrow(() -> new IllegalArgumentException("Value not present"));
    

2.6 转换值

  • 使用 map() 方法对 Optional 中的值应用一个函数。
    Optional<Integer> lengthOptional = nonNullOptional.map(String::length);
    
  • 使用 flatMap() 方法避免嵌套的 Optional 结构。
    Optional<Optional<String>> nestedOptional = Optional.of("World").map(s -> Optional.of(s));
    Optional<String> flatMapped = nestedOptional.flatMap(x -> x); // Unwraps the nested Optional
    

2.7 过滤值

  • 使用 filter() 方法根据给定条件筛选值。
    Optional<String> filteredOptional = nonNullOptional.filter(s -> s.startsWith("He"));
    

2.8. 链式调用

  • 可以组合多个操作形成链式调用来简化代码。
    Optional<String> result = Optional.of("hello")
            .filter(s -> s.length() > 3)
            .map(String::toUpperCase)
            .orElse("DEFAULT");
    

这些只是 Optional 类的一些基本用法。通过合理使用这些方法,可以使代码更加简洁、安全,并且更易于理解。


总结

Optional 是 Java 8 引入的一个重要特性,它提供了一种新的方式来处理可能为空的值,减少了空指针异常的发生,同时提升了代码的可读性和维护性。通过使用 Optional,开发者可以明确地表达出某段代码或某个方法返回值可能是空的意图,并鼓励调用者考虑如何优雅地处理这些情况。

在实际应用中,合理利用 Optional 的各种方法(如 isPresent()ifPresent()map()flatMap()filter() 等),可以让我们编写出更加健壮且易于理解的代码。尤其是当涉及到链式调用或者复杂的数据处理流程时,Optional 能够有效地简化逻辑,减少不必要的嵌套和条件判断,使得代码更加简洁明了。

然而,值得注意的是,虽然 Optional 提供了诸多便利,但它并不适合所有场景。过度使用 Optional 可能会导致代码变得冗长,特别是在基本数据类型上使用 Optional 时需要特别小心,因为这可能会带来性能上的损耗。因此,在使用 Optional 时应当根据具体情况权衡利弊,确保其带来的好处超过可能的缺点。总之,Optional 是一个强大的工具,正确地使用它可以显著提高程序的健壮性和开发效率。

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

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

相关文章

py文件打包为exe可执行文件,涉及mysql连接失败

py文件打包为exe可执行文件&#xff0c;涉及mysql连接失败 项目场景&#xff1a;使用flask框架封装算法接口&#xff0c;并使用pyinstaller打包为exe文件。使用pyinstaller打包多文件的场景&#xff0c;需要自己手动去.spec文件中添加其他文件&#xff0c;推荐使用auto-py-to-e…

Ubuntu 系统 Docker 中搭建 CUDA cuDNN 开发环境

CUDA 是 NVIDIA 推出的并行计算平台和编程模型&#xff0c;利用 GPU 多核心架构加速计算任务&#xff0c;广泛应用于深度学习、科学计算等领域。cuDNN 是基于 CUDA 的深度神经网络加速库&#xff0c;为深度学习框架提供高效卷积、池化等操作的优化实现&#xff0c;提升模型训练…

win10彻底让图标不显示在工具栏

关闭需要不显示的软件 打开 例此时我关闭了IDEA的显示 如果说只是隐藏&#xff0c;鼠标拖动一个道理 例QQ 如果说全部显示不隐藏

人脸识别和定位别的签到系统

1、功能 基于人脸识别及定位的宿舍考勤管理小程序 &#xff08;用户&#xff1a;宿舍公告、宿舍考勤查询、宿舍考勤&#xff08;人脸识别、gps 定 位&#xff09;、考勤排行、请假申请 、个人中心 管理员&#xff1a;宿舍管理、宿舍公告管理 学生信息管理、请假审批、发布宿舍…

基于YOLOv8的热力图生成与可视化:支持自定义模型与置信度阈值的多维度分析

目标检测是计算机视觉领域的重要研究方向&#xff0c;而YOLO&#xff08;You Only Look Once&#xff09;系列算法因其高效性和准确性成为该领域的代表性方法。YOLOv8作为YOLO系列的最新版本&#xff0c;在目标检测任务中表现出色。然而&#xff0c;传统的目标检测结果通常以边…

Design Compiler:库特征分析(ALIB)

相关阅读 Design Compilerhttps://blog.csdn.net/weixin_45791458/category_12738116.html?spm1001.2014.3001.5482 简介 在使用Design Compiler时&#xff0c;可以对目标逻辑库进行特征分析&#xff0c;并创建一个称为ALIB的伪库&#xff08;可以被认为是缓存&#xff09;&…

便携式雷达信号模拟器 —— 打造实战化电磁环境的新利器

在现代战争中&#xff0c;雷达信号的侦察与干扰能力直接关系到作战的成败。为了提升雷达侦察与干扰装备的实战能力&#xff0c;便携式雷达信号模拟器作为一款高性能设备应运而生&#xff0c;为雷达装备的训练、测试和科研提供了不可或缺的支持。 核心功能 便携式雷达信号模拟…

CentOS Linux升级内核kernel方法

目录 一、背景 二、准备工作 三、升级内核 一、背景 某些情况需要对Linux发行版自带的内核kernel可能版本较低&#xff0c;需要对内核kernel进行升级。例如&#xff1a;CentOS 7.x 版本的系统默认内核是3.10.0&#xff0c;该版本的内核在Kubernetes社区有很多已知的Bug&#…

【C++】多态功能细节问题分析

多态是在不同继承关系的类对象去调用同一函数&#xff0c;产生了不同的行为。值得注意的是&#xff0c;虽然多态在功能上与隐藏是类似的&#xff0c;但是还是有较大区别的&#xff0c;本文也会进行多态和隐藏的差异分析。 在继承中要构成多态的条件 1.1必须通过基类的指针或引用…

EIP-712:类型化结构化数据的哈希与签名

1. 引言 以太坊 EIP-712: 类型化结构化数据的哈希与签名&#xff0c;是一种用于对类型化结构化数据&#xff08;而不仅仅是字节串&#xff09;进行哈希和签名 的标准。 其包括&#xff1a; 编码函数正确性的理论框架&#xff0c;类似于 Solidity 结构体并兼容的结构化数据规…

基于S函数的simulink仿真

基于S函数的simulink仿真 S函数可以用计算机语言来描述动态系统。在控制系统设计中&#xff0c;S函数可以用来描述控制算法、自适应算法和模型动力学方程。 S函数中使用文本方式输入公式和方程&#xff0c;适合复杂动态系统的数学描述&#xff0c;并且在仿真过程中可以对仿真…

每日一题洛谷P8664 [蓝桥杯 2018 省 A] 付账问题c++

P8664 [蓝桥杯 2018 省 A] 付账问题 - 洛谷 (luogu.com.cn) 思路&#xff1a;要使方差小&#xff0c;那么钱不能一下付的太多&#xff0c;可以让钱少的全付玩&#xff0c;剩下还需要的钱再让钱多的付&#xff08;把钱少的补上&#xff09;。 将钱排序&#xff0c;遍历一遍&…

迅饶科技X2Modbus网关-GetUser信息泄露漏洞

免责声明&#xff1a;本号提供的网络安全信息仅供参考&#xff0c;不构成专业建议。作者不对任何由于使用本文信息而导致的直接或间接损害承担责任。如涉及侵权&#xff0c;请及时与我联系&#xff0c;我将尽快处理并删除相关内容。 漏洞描述 该漏洞的存在是由于GetUser接口在…

关于inode,dentry结合软链接及硬链接的实验

一、背景 在之前的博客 缺页异常导致的iowait打印出相关文件的绝对路径-CSDN博客 里 2.2.3 一节里&#xff0c;我们讲到了file&#xff0c;fd&#xff0c;inode&#xff0c;dentry&#xff0c;super_block这几个概念&#xff0c;在这篇博客里&#xff0c;我们针对inode和dentr…

PandasAI:当数据分析遇上自然语言处理

数据科学的新范式 在数据爆炸的时代&#xff0c;传统的数据分析工具正面临着前所未有的挑战。数据科学家们常常需要花费70%的时间在数据清洗和探索上&#xff0c;而真正的价值创造时间却被大幅压缩。PandasAI的出现&#xff0c;正在改变这一现状——它将生成式AI的强大能力注入…

Unity网络开发基础 (3) Socket入门 TCP同步连接 与 简单封装练习

本文章不作任何商业用途 仅作学习与交流 教程来自Unity唐老狮 关于练习题部分是我观看教程之后自己实现 所以和老师写法可能不太一样 唐老师说掌握其基本思路即可,因为前端程序一般不需要去写后端逻辑 1.认识Socket的重要API Socket是什么 Socket&#xff08;套接字&#xff0…

做题记录:和为K的子数组

来自leetcode 560 前言 自己只会暴力&#xff0c;这里就是记录一下前缀和哈希表的做法&#xff0c;来自灵神的前缀和哈希表&#xff1a;从两次遍历到一次遍历&#xff0c;附变形题 正文 首先&#xff0c;这道题无法使用滑动窗口&#xff0c;因为滑动窗口需要满足单调性&am…

VMware虚拟机卡顿、CPU利用率低、编译Linux内核慢,问题解决与实验对比

目录 一、总结在前面&#xff08;节约时间就只看这里&#xff09;0 环境说明1 遇到的问题&#xff1a;2 问题的原因&#xff1a;3 解决办法&#xff1a;4 实验验证&#xff1a;5 关于虚拟机内核数量设置6 关于强行指定Vm能用的CPU内核 二、管理员启动&#xff0c;实验对比实验1…

【7】数据结构的队列篇章

目录标题 队列的定义顺序队列的实现初始化入队出队顺序队列总代码与调试 循环队列的实现初始化入队出队获取队首元素循环队列总代码与调试 链式队列的实现链式队列的初始化入队出队获取队首元素链式队列总代码与调试 队列的定义 定义&#xff1a;队列&#xff08;Queue&#x…

颜色归一化操作

当我们不太关注图像具体细节&#xff0c;只关注图像大致的内容时&#xff0c;为了避免光照角度、光照强度对图像的影响&#xff0c;可以采用下面进行归一化操作。这种颜色系统具有通道对表面方向、照明方向具有鲁棒性的特性&#xff0c;适用于图像分割等领域&#xff0c;在机器…