【Optional】告别丑陋判空,使用Optional类
一、概述
当项目中充斥着大量的、丑陋的判空语句,如下:
if (user != null) {Address address = user.getAddress();if (address != null) {Country country = address.getCountry();if (country != null) {String isocode = country.getIsocode();if (isocode != null) {isocode = isocode.toUpperCase();}}}
}
那怎么办呢?使用Optional优化后:
String result = Optional.ofNullable(user).map(u -> u.getAddress()).map(a -> a.getCountry()).map(c -> c.getIsocode()).orElse("default");
二、创建Optional实例
Optional类, 是对value值进行了包装,它的值可能是null, 也可能不是null,一共有两个方法创建Optional实例
2.1 static Optional of(T value)
@Test
public void testOf() {Integer value = 20;// 正常执行Optional<Integer> op = Optional.of(value);value = null;// 报空指针op = Optional.of(value);
}
2.2 static Optional ofNullable(T value)
@Testpublic void testOfNullable() {Integer value = 2;// 正常Optional<Integer> op = Optional.ofNullable(value);value = null;// 也正常 不报错op = Optional.ofNullable(value);}
2.3源码分析
empty()的作用就是返回EMPTY对象。
相比较of(T value)
的区别就是,当value值为null时,of(T value)
会报NullPointerException异常;ofNullable(T value)
不会throw Exception,ofNullable(T value)
直接返回一个EMPTY对象。
三、获取Optional中的值
3.1T get()
获取原生value的方法,如果value是空,则直接抛出异常,否则返回。
@Test
public void testGet() {Integer value = 2;// 正常Optional<Integer> op = Optional.ofNullable(value);Integer opVal = op.get();Assert.assertEquals(opVal, value);op = Optional.ofNullable(null);// 抛出异常op.get();
}
源码:
3.2T orElse(T other)
如果值存在返回,否则返回orElse中传入的other
@Test
public void testOrElse() {// 正常Optional<Integer> op = Optional.ofNullable(2);Integer opVal = op.orElse(3);Assert.assertEquals(opVal, new Integer(2));op = Optional.ofNullable(null);// 为空,则返回3opVal = op.orElse(3);Assert.assertEquals(opVal, new Integer(3));
}
源码:
3.3T orElseGet(Supplier<? extends T> other)
如果存在则返回该值,否则调用other这个函数编程并返回该调用的结果。
@Test
public void testOrElseGet() {// 正常Optional<Integer> op = Optional.ofNullable(2);Integer opVal = op.orElseGet(()-> {return new Integer(3);});Assert.assertEquals(opVal, new Integer(2));op = Optional.ofNullable(null);// 如果为空,则返回3opVal = op.orElseGet(()-> {return new Integer(3);});Assert.assertEquals(opVal, new Integer(3));
}
源码:
3.4orElseGet和orElse有什么区别吗?
orElse()
和 orElseGet()
的不同之处在于当 ofNullable()
传入参数不为空时,orElse()
方法仍然创建了 other这个 对象。与之相反,orElseGet()
方法不创建对象。在执行较密集的调用时,比如调用 Web 服务或数据查询,这个差异会对性能产生重大影响。
而且我们还可以在orElseGet
方法中加些日志,可以把这种为空的异常情况暴露出来。
3.5T orElseThrow(Supplier<? extends X> exceptionSupplier)
如果存在则返回该值,否则为空的话可以抛出自定义的异常
@Test
public void testOrElseThrow() {Optional<Integer> op = Optional.ofNullable(null);// 为空,则抛出指定的异常类型Integer opVal = op.orElseThrow(()-> {return new RuntimeException();});// 或抛出runtime异常Assert.assertEquals(opVal, new Integer(3));
}
源码:
四、判断Optional是否为空
4.1boolean isPresent()
判断value是否为空
@Test
public void testIsPresent() {Optional<Integer> op = Optional.ofNullable(null);// 为空Assert.assertFalse(op.isPresent());
}
4.2void ifPresent(Consumer<? super T> consumer)
如果存在值,则调用对应的consumer方法,否则不执行任何操作。
@Test
public void testIfPresent() {Optional<Integer> op = Optional.ofNullable(5);op.ifPresent(value -> {// 如果存在,打印出来System.out.println(value);});
}
五、Optional中的过滤、转换方法
5.1Optional filter(Predicate<? super T> predicate)
对Optional中的value进行过滤,如果不匹配,返回空
@Test
public void testFilter() {// 不满足过滤条件,返回空 OptionalOptional<String> op = Optional.ofNullable("10").filter(item -> "15".equals(item));Assert.assertFalse(op.isPresent());
}
5.2Optional map(Function<? super T, ? extends U> mapper)
对Optional中的value进行转换映射为另外一个对象,如果value为空,返回empty Optional
@Testpublic void testMap(){Optional<Optional<Integer>> integer = Optional.ofNullable("5").map(item -> Optional.of(Integer.valueOf(item)));}
源码:
5.3Optional flatMap(Function<? super T, Optional> mapper)
接受一个返回值为Optional的映射函数参数,该返回值亦是flatMap方法的返回值若结果为空,则返回 空Optional。它也map的区别,我们用一个例子演示出来。
@Testpublic void testflatMap(){Optional<Integer> integer = Optional.ofNullable("5").flatMap(item -> Optional.of(Integer.valueOf(item)));}
源码:
小结: 如果对于返回值非Optional类型,可以用map方法, 否则使用flatMap更加方便
六、实战案例
6.1例一
以前写法:
public String getCity(User user) throws Exception{if(user!=null){if(user.getAddress()!=null){Address address = user.getAddress();if(address.getCity()!=null){return address.getCity();}}}throw new Excpetion("取值错误");}
JAVA8写法:
public String getCity(User user) throws Exception{return Optional.ofNullable(user).map(u-> u.getAddress()).map(a->a.getCity()).orElseThrow(()->new Exception("取指错误"));
}
6.2例二
以前写法:
if(user!=null){dosomething(user);
}
JAVA8写法:
Optional.ofNullable(user).ifPresent(u->{dosomething(u);
});
6.3例三
以前写法:
public User getUser(User user) throws Exception{if(user!=null){String name = user.getName();if("zhangsan".equals(name)){return user;}}else{user = new User();user.setName("zhangsan");return user;}
}
JAVA8写法:
public User getUser(User user) {return Optional.ofNullable(user).filter(u->"zhangsan".equals(u.getName())).orElseGet(()-> {User user1 = new User();user1.setName("zhangsan");return user1;});
}
相关文章:

【Optional】告别丑陋判空,使用Optional类
一、概述 当项目中充斥着大量的、丑陋的判空语句,如下: if (user ! null) {Address address user.getAddress();if (address ! null) {Country country address.getCountry();if (country ! null) {String isocode country.getIsocode();if (isocod…...

魔兽世界服务端端新手搭建教程
明杰也是很久以前开始研究魔兽服务器架设,主要原因是亚服经常要排队6-7个小时,去不排除的服和单机没啥区别,以怀旧服玩到10级以后就开始玩335端,一开始也和新入手的人一样云里雾里的,经过长时间的学习总算有点成就,向新…...

宝塔搭建实战人才求职管理系统mobile手机端vue源码(五)
大家好啊,我是测评君,欢迎来到web测评。 上一期给大家分享骑士cms会员管理member前端vue在本地运行打包、宝塔发布部署的方式,本期给大家分享,mobile移动端vue怎么在本地运行,打包,实现线上功能更新替换的方…...
生态应用:探讨 NGINX 与上下游系统集成时的开发经验
NGINX 作为一款高性能的 Web 服务器和反向代理服务器,在各种应用场景中广泛应用。随着业务的发展,我们在使用 NGINX 时,可能需要将其与其他系统进行集成,以实现更加复杂的业务需求。 本文将结合实际应用场景,探讨 NGI…...

ArcGIS批量拼接大量栅格遥感影像:Mosaic工具
本文介绍在ArcGIS下属的ArcMap软件中,基于Mosaic工具,批量对大量栅格遥感影像文件加以拼接、镶嵌的方法。 在GIS应用中,我们时常需要对大量栅格遥感影像数据加以批量拼接的工作。这一步骤可以基于Python语言实现,具体可以参考文章…...
Flink UI部署jar包报错
错误描述: 通过Flink的UI中的Submit New Job菜单添加jar包的时候提示报错。报错信息的关键字是“The LocalStreamEnvironment cannot be used when submitting a program through a client, or running in a TestEnvironment context”,最关键的是“Loc…...
Linux就该这么学:RAID与LVM磁盘阵列技术
这里写目录标题 7.1.1 部署磁盘阵列7.1.2 损坏磁盘阵列及修复7.1.3 磁盘阵列+备份盘7.1.4 删除磁盘阵列1. 需要将所有的磁盘都设置成停用状态:2. 逐一移除出去3. 续停用整个RAID磁盘阵列7.2 LVM逻辑卷管理器7.2.1 部署逻辑卷7.2.2 扩容逻辑卷7.2.3 缩小逻辑卷7.2.4 逻辑卷快照…...

Prometheus+Grafana监控
1、简介1.1 Prometheus官网地址:https://prometheus.io/Prometheus是一个开源的监控系统,起源于SoundCloud。它由以下几个核心组件构成:数据爬虫: 根据配置的时间定期的通过HTTP抓去metrics数据。time-series 数据库: …...
【JUC2022】第三章 线程中断与 LockSupport
【JUC2022】第三章 线程中断与 LockSupport 文章目录【JUC2022】第三章 线程中断与 LockSupport一、线程中断1.什么是中断机制2.中断 API3.代码实现4.Thread.sleep()二、LockSupport1.什么是 LockSupport2.代码实现3.总结一、线程中断 1.什么是中断机制 首先,一个…...
数据结构刷题(七):202快乐数、1两数之和、454四数相加II、15三数之和、18四树之和
1.快乐数题目链接思路:使用set,当set中出现相同元素时,返回false注意:while循环中语句顺序; 除数取余。解法:public boolean isHappy(int n){Set<Integer> res new HashSet<>();int[] arr ne…...
华为机试题:HJ80 整型数组合并(python)
文章目录知识点详解1、input():获取控制台(任意形式)的输入。输出均为字符串类型。 1.1、int(input()) 与 map(int, input().spilt()) 的区别 1.2、input() 与 list(input()) 的区别、及其相互转换方法2、print() :打印输出…...

spring boot——自定义依赖实现自动配置
需求 要实现的功能是:实现一个可以支持miniooss两种方式,上传下载文件的自定义依赖。其中还包括一些创建桶、删除桶、删除文件等功能,但是最主要的是实现自动配置。 如果对spring理解很深的话,自动配置这些东西很容易理解&#…...

QMap 判断是否value是否已经存在,结合Sleep函数测试
网上查了资料,基本说的都是通过.value判断是否已经之前的key值,但是尝试.了一下发现有.key的函数,对比着来就感觉这个函数是用来判断是否已经存在value值,于是开始百度也几乎没有找到相关资料,只好自己看官方文档&…...

vue后台管理系统项目-table选择多行数据分页列表、一键全选重置功能
table选择多行数据 功能介绍: 1.列表分页功能; 2.一键全选,选中列表所有数据; 3.全选,选中当前页数据; 4.重置,清除选中状态; 5.列表搜索查询; 效果: 1.列表分…...

论文解读 | [CVPR2019] 基于自适应文本区域表示的任意形状场景文本检测
目录 1 研究背景及意义 2 总体设计 3 方法论 3.1 自适应文本区域表示 3.2 文本建议 3.3 建议改进 4 损失函数 5 实验及结果 1 研究背景及意义 现有的场景文本检测方法使用固定点数的多边形来 表示文本区域。例如,水平文本使用2个点(左上/右下)表示文本区域&…...

2月编程语言排行榜谁还没有看?
近日,TIOBE公布了2023年2月编程语言排行榜,本月各个语言表现如何?谁又摘得桂冠?一起来看看吧! TIOBE 2月Top15编程语言: 详细榜单查看TIOBE官网 https://www.tiobe.com/tiobe-index/ 关注IT行业的小伙伴们…...

nginx.conf配置方法详细介绍
从前面的内容学习中,我们知道Nginx的核心配置文件默认是放在/usr/local/nginx/conf/nginx.conf,这一节,我们就来学习下nginx.conf的内容和基本配置方法。读取Nginx自带的Nginx配置文件,我们将其中的注释部分【学习一个技术点就是在…...

【微信小程序】一文带你吃透开发中的常用组件
写在前面 小程序中的组件也是由宿主环境提供的,开发者可以基于组件快速搭建出漂亮的页面结构。 官方把小程序的组件分为了9大类,分别是: 1.视图容器 2.基础内容 3.表单组件 4.导航组件 5.媒体组件 6.地图组件 7.画布组件 …...

Nginx 部署 Vue 项目以及 Vue 项目刷新出现 404 的问题(完整步骤)(亲测有效)
Nginx 部署 Vue 项目以及 Vue 项目刷新出现 404 的问题(完整步骤)(亲测有效) 1.流程步骤(本教程下载的是1.20.2版本,放在D盘) 1-1. 首先去官方下载 nginx ,然后在当前目录下创建ht…...

leaflet 加载geojson数据,随机显示不同颜色的circleMarker
第086个 点击查看专栏目录 本示例的目的是介绍演示如何在vue+leaflet项目中加载geojson数据,随机显示不同颜色的circleMarker. 直接复制下面的 vue+leaflet源代码,操作2分钟即可运行实现效果 文章目录 示例效果配置方式示例源代码(共89行)相关API专栏目标示例效果 配置方式…...
Windows 下搭建 Zephyr 开发环境
1. 系统要求 操作系统:Windows 10/11(64位)磁盘空间:至少 8GB 可用空间(Zephyr 及其工具链较大)权限:管理员权限(部分工具需要) 2. 安装必要工具 winget安装依赖工具&am…...
PCB特种工艺应用扩展:厚铜、高频与软硬结合板
新能源汽车与消费电子驱动PCB特种工艺创新,厚铜板降阻30%,软硬结合板渗透率年增15%。 1. 厚铜板:新能源高压平台核心 技术突破:猎板PCB量产10oz厚铜板(传统为3oz),载流能力提升200%,…...
Spring中循环依赖问题的解决机制总结
一、解决机制 1. 什么是循环依赖 循环依赖是指两个或多个Bean之间相互依赖对方,形成一个闭环的依赖关系。最常见的情况是当Bean A依赖Bean B,而Bean B又依赖Bean A时,就形成了循环依赖。在Spring容器初始化过程中,如果不加以特殊…...

60天python训练计划----day45
DAY 45 Tensorboard使用介绍 知识点回顾: tensorboard的发展历史和原理tensorboard的常见操作tensorboard在cifar上的实战:MLP和CNN模型 之前的内容中,我们在神经网络训练中,为了帮助自己理解,借用了很多的组件&#x…...
《前端面试题:CSS3新特性》
CSS3新特性指南:从基础到实战详解 CSS3作为现代Web开发的核心样式标准,彻底改变了前端开发者的工作方式。它不仅解决了传统CSS的诸多痛点,还引入了强大的布局模型、动画系统和响应式设计能力。本文将全面解析CSS3的十大核心新特性࿰…...
C语言的全称:(25/6/6)
C语言,全称为"C Programming Language"(C程序设计语言),是一种广泛使用的计算机编程语言。它是由Dennis Ritchie于1972年在贝尔实验室设计的,继承了B语言的许多思想,并加入了数据类型的概念及其他…...

Excel处理控件Aspose.Cells教程:使用 C# 在 Excel 中创建组合图表
可视化项目时间线对于有效规划和跟踪至关重要。在本篇教程中,您将学习如何使用 C# 在 Excel 中创建组合图。只需几行代码,即可自动生成动态、美观的组合图。无论您是在构建项目管理工具还是处理内部报告,本指南都将向您展示如何将任务数据转换…...

Day46 Python打卡训练营
知识点回顾: 1. 不同CNN层的特征图:不同通道的特征图 2. 什么是注意力:注意力家族,类似于动物园,都是不同的模块,好不好试了才知道。 3. 通道注意力:模型的定义和插入的位置 4. 通道注意力后…...
如何选择专业数据可视化开发工具?为您拆解捷码全功能和落地指南!
分享大纲: 1、捷码核心功能:4维能力支撑大屏开发 2、3步上手:可视化大屏开发操作路径 3、适配场景:8大行业已验证方案 在各行各业要求数字化转型时代,数据可视化大屏已成为众多企业数据驱动的核心工具。面对市场上繁杂…...
Python实现快速排序的三种经典写法及算法解析
今天想熟悉一下python的基础写法,那就从最经典的快速排序来开始吧: 1、经典分治写法(原地排序) 时间复杂度:平均O(nlogn),最坏O(n) 空间复杂度:O(logn)递归栈空间 特点:通过左右指针…...