Java中关于Optional的 orElse 操作,以及 orElse 与 orElseGet 的区别

文章目录

      • 1. 大概说明
      • 2. 详细分析
        • 2.1 .orElse 操作
        • 2.2 .orElse 的作用:避免空指针异常
        • 2.3 为什么要用?
        • 2.4 orElseGet如何使用
        • 2.5 orElse和orElseGet的区别

1. 大概说明

这篇文章的目的是为了说明:

  • orElse 如何使用
  • orElseGet 如何使用
  • 两者的区别

备注:orElse 可能导致 NullPointerException,当 orElse 的参数是间接计算得来的时候。虽然这种说法有点牵强(因为并不是orElse导致了空指针异常),但是使用 orElseGet 确实可以避免这种情况。

2. 详细分析

2.1 .orElse 操作

先看个例子:

Optional<String> optional = Optional.empty();
System.out.println(optional.isPresent); // ---->输出:false
// 返回 false,表示 Optional 里没有值。

解析:

  • Optional.empty() 代表 一个空的 Optional 实例,即 Optional 没有值
  • Optional.ofNullable(null) 等价于 Optional.empty()
2.2 .orElse 的作用:避免空指针异常
  • 如果 Optional 为空(即 Optional.empty()),就返回 orElse() 里提供的默认值。
  • 如果 Optional 有值,就直接返回这个值,不执行 orElse() 提供的默认值。

注意:不管 Optional 为不为空,这个值都会创建,只不过为空时,才使用

// 举例:当Optional为空时,orElse()才会触发
String result = Optional.ofNullable(null).orElse("默认值");
System.out.println(result); // ---->输出:默认值
// 再举个例子:
Map<Long, List<String>> map = new HashMap<>();
map.put(1L, Arrays.asList("A", "B", "C"));List<String> result = Optional.ofNullable(map.get(1L)).orElse(Collections.emptyList()); // 为null就返回 空列表
System.out.println(result);  // ---->输出:[A, B, C] (orElse() 没起作用)
2.3 为什么要用?
List<String> result = Optional.ofNullable(map.get(0L)).orElse(Collections.emptyList()).stream();

如果 map.get(0L) == null,那么:

  • 不加 .orElse(Collections.emptyList()).stream() 会报 NullPointerException
  • 加了 .orElse(Collections.emptyList()).stream() 能正常执行 ✅
2.4 orElseGet如何使用

再来看看 orElseGet 中如何使用:

orElseGet 作用:获取数据并且设置数据为空时的默认值。如果数据不为空就能获取到该数据;如果为空则返回传入的参数来创建对象。

具体的使用案例可看下图:

在这里插入图片描述在这里插入图片描述

2.5 orElse和orElseGet的区别

orElse()orElseGet() 都是Optional 类中的方法,用于在 Optional 为空时提供默认值。但它们的区别在于 默认值的获取方式

1、orElse(T other)

  • 直接传递一个默认值
  • 即使 Optional里有值,也会创建 other 对象,但不会使用它
  • 适用于默认值创建代价较低的情况

2、orElseGet(Supplier<? extends T> supplier)

  • 传递的是一个 Supplier(懒加载:只有需要的时候才会创建)接口,它是一个函数式接口,形式是这样的:()->{ return computedResult },即入参为空,有返回值(任意类型的)
  • 仅当 Optional 为空时才会执行 supplier.get(),不会提前创建默认值
  • 适用于默认值创建代价较高的情况

看个例子:

class User {// 中文名private String chineseName;// 英文名private EnglishName englishName;
}class EnglishName {// 全名private String fullName;// 简写private String shortName;
}

假如我们现在有 User 类,用户注册账号时,需要提供自己的中文名或英文名,或都提供,我们抽象出一个EnglishName 类,它包含英文名的全名和简写(因为有的英文名确实太长了)。现在,我们希望有一个User.getName() 方法,它可以像下面这样实现:

class User {// ... 之前的内容public String getName1() {return Optional.ofNullable(chineseName).orElse(englishName.getShortName());}public String getName2() {return Optional.ofNullable(chineseName).orElseGet(() -> englishName.getShortName());}
}

写了两个版本,分别使用 orElse 和 orElseGet。现在,你可以看出 getName1() 方法有什么风险了吗?它会出现空指针异常吗?----> 是的。当用户只提供了中文名时,此时 englishName 属性是 null,但是在 orElse 中,englishName.getShortName()总是会执行。而在 getName2() 中,这个风险却没有。

再举个例子:

public class Test {public static void main(String[] args) {System.out.println("orElse() 的情况:");String result1 = Optional.ofNullable("实际值").orElse(test2());System.out.println("最终结果: " + result1);System.out.println("orElseGet() 的情况:");String result2 = Optional.ofNullable("实际值").orElseGet(() -> test2());System.out.println("最终结果: " + result2);}public static String test2() {System.out.println("执行昂贵的计算...");return "昂贵默认值";}
}

输出:

orElse() 的情况:
执行昂贵的计算...
最终结果: 实际值orElseGet() 的情况:
最终结果: 实际值

至此,文章结束,记得点赞收藏评论!!!

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

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

相关文章

数据结构-树(详解)

目录 一、树的基本概念二、树的节点结构三、树的基本操作&#xff08;一&#xff09;插入操作&#xff08;二&#xff09;删除操作&#xff08;三&#xff09;查找操作&#xff08;四&#xff09;遍历操作 四、树的实现五、总结 一、树的基本概念 树是一种非线性数据结构&…

【eNSP实战】配置端口映射(NAT Server)

拓图 要求&#xff1a; 将AR1上的GE 0/0/1接口的地址从TCP协议的80端口映射到内网 Web服务器80端口 AR1接口配置 interface GigabitEthernet0/0/0ip address 192.168.0.1 255.255.255.0 # interface GigabitEthernet0/0/1ip address 11.0.1.1 255.255.255.0 # ip route-s…

RabbitMQ 基本原理详解

1. 引言 在现代分布式系统中&#xff0c;消息队列&#xff08;Message Queue&#xff09;是实现异步通信、解耦系统组件、提高系统可靠性和扩展性的重要工具。RabbitMQ 作为一款开源的消息中间件&#xff0c;因其高性能、易用性和丰富的功能&#xff0c;被广泛应用于各种场景。…

算法——层序遍历和中序遍历构造二叉树

晴问 #include <iostream> #include <vector> #include <queue> #include <unordered_map>using namespace std;struct TreeNode {int data;TreeNode *left;TreeNode *right;TreeNode(int data) : data(data), left(nullptr), right(nullptr) {} };//…

prometheus自定义监控(pushgateway和blackbox)和远端存储VictoriaMetrics

1 pushgateway采集 1.1 自定义采集键值 如果自定义采集需求时&#xff0c;就可以通过写脚本 定时任务定期发送数据到 pushgateway 达到自定义监控 1.部署 pushgateway&#xff0c;以 10.0.0.42 节点为例 1.下载组件 wget https://github.com/prometheus/pushgateway/relea…

feign配置重试次数不生效

一、问题产生 自定义重试次数&#xff0c;实现如下 ConditionalOnProperty(prefix "feign.client", name "enable", havingValue "true") Configuration public class FeignConfig {Beanpublic FeignInterceptor feignInterceptor() {retur…

Dify使用部署与应用实践

最近在研究AI Agent&#xff0c;发现大家都在用Dify&#xff0c;但Dify部署起来总是面临各种问题&#xff0c;而且我在部署和应用测试过程中也都遇到了&#xff0c;因此记录如下&#xff0c;供大家参考。Dify总体来说比较灵活&#xff0c;扩展性比较强&#xff0c;适合基于它做…

二叉树的统一迭代法 标记法

我们以中序遍历为例&#xff0c;在二叉树&#xff1a;听说递归能做的&#xff0c;栈也能做&#xff01; (opens new window)中提到说使用栈的话&#xff0c;无法同时解决访问节点&#xff08;遍历节点&#xff09;和处理节点&#xff08;将元素放进结果集&#xff09;不一致的情…

BaseActivity 和 BaseFragment 的现代化架构:ViewBinding 与 ViewModel 的深度整合

BaseActivity 和 BaseFragment 实现&#xff0c;集成了 View Binding&#xff0c;并增加了对 Lifecycle 和 ViewModel 的支持&#xff0c;同时进一步简化了代码结构&#xff0c;使其更易用、更灵活。 启用 View Binding 确保在 build.gradle 中启用了 View Binding&#xff1a…

从零开始学习机器人---如何高效学习机械原理

如何高效学习机械原理 1. 理解课程的核心概念2. 结合图形和模型学习3. 掌握公式和计算方法4. 理论与实践相结合5. 总结和复习6. 保持好奇心和探索精神 总结 机械原理是一门理论性和实践性都很强的课程&#xff0c;涉及到机械系统的运动、动力传递、机构设计等内容。快速学习机械…

剖析sentinel的限流和熔断

sentinel的限流和熔断 前言源码分析滑动窗口源码限流源码熔断源码 完结撒花&#xff0c;sentinel源码还是挺简单的&#xff0c;如有需要收藏的看官&#xff0c;顺便也用发财的小手点点赞哈&#xff0c;如有错漏&#xff0c;也欢迎各位在评论区评论&#xff01; 前言 平时发起一…

硬盘分区误删后的数据救赎

一、硬盘分区误删的概述 硬盘分区误删&#xff0c;是许多电脑用户在使用过程中可能遭遇的棘手问题。分区&#xff0c;作为硬盘上存储数据的逻辑单元&#xff0c;一旦被误删除&#xff0c;不仅会导致该分区内的所有数据瞬间消失&#xff0c;还可能影响到整个硬盘的存储结构和数…

代码随想录算法训练营第三十五天(20250303) |01背包问题 二维,01背包问题 一维,416. 分割等和子集 -[补卡20250316]

01背包问题 二维 链接 遍历物品没有大小顺序要求重点是模拟&#xff0c;推导出递推公式 #include <iostream> #include <vector>int main(){int m, n;std::cin>>m>>n;std::vector<int> weight(m,0),value(m,0);for(int i{0}; i<m; i){std:…

老牌软件,方便处理图片,量大管饱。

今天介绍的图片查看器名字是&#xff1a;FastStone Image Viewer&#xff0c;是一款可查看、编辑、批量重命名、批量转换的图片查看软件。文末有分享链接。 软件以资源管理器的方式管理你电脑里的图片&#xff0c;点击左侧可选择文件夹&#xff0c;右边可预览图片。 软妹用得最…

【数据库相关】mysql数据库巡检

mysql数据库巡检 巡检步骤**一、基础状态检查****二、服务器资源监控****CPU使用****内存使用****磁盘I/O****网络流量** **三、数据库内部健康度****全局状态****慢查询监控****锁与并发** **四、存储引擎健康****InnoDB引擎****MyISAM引擎** **五、日志与备份****六、安全与权…

Python进阶编程总结

&#x1f9d1; 博主简介&#xff1a;CSDN博客专家&#xff0c;历代文学网&#xff08;PC端可以访问&#xff1a;https://literature.sinhy.com/#/literature?__c1000&#xff0c;移动端可微信小程序搜索“历代文学”&#xff09;总架构师&#xff0c;15年工作经验&#xff0c;…

Redis复制(replica)主从模式

Redis主从复制 Redis 的复制&#xff08;replication&#xff09;功能允许用户根据一个 Redis 服务器来创建任意多个该服务器的复制品&#xff0c;其中被复制的服务器为主服务器&#xff08;master&#xff09;&#xff0c;而通过复制创建出来的服务器复制品则为从服务器&#…

Adobe Premiere Pro2023配置要求

Windows 系统 最低配置 处理器&#xff1a;Intel 第六代或更新版本的 CPU&#xff0c;或 AMD Ryzen™ 1000 系列或更新版本的 CPU&#xff0c;需要支持 Advanced Vector Extensions 2&#xff08;AVX2&#xff09;。操作系统&#xff1a;Windows 10&#xff08;64 位&#xff…

【Kubernets】Deployment 和 StatefulSet 有什么区别?什么时候用 StatefulSet?

Deployment 和 StatefulSet 的区别 在 Kubernetes 中&#xff0c;Deployment 和 StatefulSet 都用于管理 Pod&#xff0c;但它们适用于不同的场景。 1. Deployment&#xff1a;管理无状态应用 特点&#xff1a; 无状态&#xff1a;Pod 之间相互独立&#xff0c;不需要保持顺…

R语言零基础系列教程-03-RStudio界面介绍与关键设置

代码、讲义、软件回复【R语言03】获取。 设置位置: 菜单栏 - Tools - Blobal Options 设置 通用设置 设置面板左侧General选项 版本选择: 一般只用一个版本即可 默认工作目录设置: 你希望RStudio打开时是基于哪个目录进行工作可以不设置, 因为脚本一般都是放置在特定项目路…