告别空指针让代码变优雅,Optional使用图文例子源码解读
一、前言
我们在开发中最常见的异常就是NullPointerException
,防不胜防啊,相信大家肯定被坑过!
这种基本出现在获取数据库信息中、三方接口,获取的对象为空,再去get出现!
解决方案当然简单,只需要判断一下,不是空在去后续操作,为空返回!
所有在JDK8时出现了专门处理的方案,出来很早了,但是小编惭愧一直没有去使用它!
最近在看《Java开发手册》,一直想着提高自己的代码水平,文中就指出了使用Optional
来解决NullPointerException
!
二、Java开发手册规范
小编使用的是2022版的黄山版,29页写到:
【推荐】
防止 NPE,是程序员的基本修养,注意 NPE 产生的场景:
- 返回类型为基本数据类型,return 包装数据类型的对象时,自动拆箱有可能产生 NPE
反例:public int method() { return Integer 对象; },如果为 null,自动解箱抛 NPE。
- 数据库的查询结果可能为 null。
- 集合里的元素即使 isNotEmpty,取出的数据元素也可能为 null。
- 远程调用返回对象时,一律要求进行空指针判断,防止 NPE。
- 对于 Session 中获取的数据,建议进行 NPE 检查,避免空指针。
- 级联调用 obj.getA().getB().getC();一连串调用,易产生 NPE。
正例:使用 JDK8 的 Optional
类来防止 NPE 问题。
这份手册还是不错的,推荐反复阅读,虽然进不去大厂,也要自觉约束自己的代码风格,努力向大厂靠!
大家现在不知道哪里找的可以下载一下:
《Java开发手册》
三、Optional常用方法
小编带大家一起从api文档中的方法,一个个带大家慢慢去了解它!
1. empty()
返回一个空的Optional实例:Optional.empty
Optional<Object> empty = Optional.empty();
log.info("empty值:{}",empty);
2. of(T value)
传入一个参数,返回一个
Optional
对象,如果参数为空,报NullPointerException
!
Test testNew = new Test();
Test test = null;
Optional<Test> optionalNew = Optional.of(testNew);
log.info(" optional对象:{}",optionalNew);
Optional<Test> optional = Optional.of(test);
源码查看:
我们看到参数为空会报NullPointerException
,我们去方法内部看一下就明白了:
public static <T> Optional<T> of(T value) {return new Optional<>(value);
}
private Optional(T value) {this.value = Objects.requireNonNull(value);
}
public static <T> T requireNonNull(T obj) {if (obj == null)throw new NullPointerException();return obj;
}
我们发现是在Objects
类中的requireNonNull
方法中判断了是否为空!
这个还会出现NullPointerException
,所以我们一般使用下面的这个方法!
3. ofNullable(T value)
参数传入一个对象,返回一个Optional对象,如果为空,将返回一个空的Optional对象,就等于Optional.empty
Test testNew = new Test();
Test test = null;
Optional<Test> optionalNew = Optional.of(testNew);
log.info(" optional对象:{}",optionalNew);Optional<Test> optionalTest = Optional.ofNullable(test);
log.info(" optional对象中的ofNullable方法返回值:{}",optionalTest);
Optional<Test> optionalTestNew = Optional.ofNullable(testNew);
log.info(" optional对象中的ofNullable方法new返回值:{}",optionalTestNew);
源码查看:
我们发现是在方法开始进行非空判断,再去调用上面的of(T value)
方法
public static <T> Optional<T> ofNullable(T value) {return value == null ? empty() : of(value);
}
4. get()
如果此Optional中存在值,则返回该值,否则抛出
NoSuchElementException
。
Test testNew = new Test();
Test test = null;
Optional<Test> optionalNew = Optional.of(testNew);
log.info(" optional对象:{}",optionalNew);
// Optional<Test> optional = Optional.of(test);Optional<Test> optionalTest = Optional.ofNullable(test);
log.info(" optional对象中的ofNullable方法返回值:{}",optionalTest);
Optional<Test> optionalTestNew = Optional.ofNullable(testNew);
log.info(" optional对象中的ofNullable方法new返回值:{}",optionalTestNew);Test test2 = optionalTestNew.get();
log.info("原来有值的:经过Optional包装后get后得到原来的值:{}",test2);
Test test1 = optionalTest.get();
log.info("原来没有值的:经过Optional包装后get后得到原来的值:{}",test1);
源码查看:
调用开始会进行值判断,如果为空则抛异常!
public T get() {if (value == null) {throw new NoSuchElementException("No value present");}return value;
}
5. isPresent()
如果存在值,则返回true,否则返回false。
这里代码就不加上面的,大家参考上面的获取一个Optional对象
boolean present = optionalTestNew.isPresent();
log.info("optionalTestNew调用是否为空:{}",present);
boolean present1 = optionalTest.isPresent();
log.info("optionalTest调用是否为空:{}",present1);
源码查看:
这就比较简单了!
public boolean isPresent() {return value != null;
}
6. ifPresent(Consumer<? super T> consumer)
如果存在值,则使用该值调用指定的使用者,否则不执行任何操作。
主要的就是入参数一个函数式接口,有值就会去执行,为空则不进行任何操作!
小技巧:
开始对lambda不了解时,可以先按照上面这种方式进行写,
大家可以看到Idea给置灰了,就是可以优化,我们Alt+Enter
然后再次Enter
就会变成后面的lambda!
optionalTest.ifPresent(new Consumer<Test>() {@Overridepublic void accept(Test test) {log.info("我是调用ifPresent执行后的打印=====");}
});
optionalTestNew.ifPresent(testInner -> log.info("我是调用ifPresent执行后的打印"));
源码查看:
还是先判断不为空才去执行函数式接口!
public void ifPresent(Consumer<? super T> consumer) {if (value != null)consumer.accept(value);
}
7. filter(Predicate<? super T> predicate)
如果存在值,并且该值符合规则,则返回描述该值的Optional,否则返回空Optional
是一个Predicate函数接口,可以传入实现了Predicate接口的lambda表达式!
如果不符合条件就会返回一个Optional.empty
testNew.setName("萧炎");
testNew.setAge(33);
Optional<Test> optionalTest1 = optionalTestNew.filter(test1 -> test1.getAge() > 30);
log.info("过滤后的结果:{}",optionalTest1.get());
源码查看:
就是判断一下表达式和值是否为空,然后就是根据规则判断
public Optional<T> filter(Predicate<? super T> predicate) {Objects.requireNonNull(predicate);if (!isPresent())return this;elsereturn predicate.test(value) ? this : empty();
}
8. map(Function<? super T,? extends U> mapper)
如果存在值,则将提供的映射函数应用于该值,如果结果为非空,则返回描述结果的Optional。否则,返回空的Optional。
也是一个函数式接口!
Optional<String> stringOptional = optionalTestNew.map(Test::getName);
log.info("map后获得字段值:{}",stringOptional.get());
源码查看:
也是进行非空判断,然后执行lambda得到字段后放到ofNullable方法中!
public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {Objects.requireNonNull(mapper);if (!isPresent())return empty();else {return Optional.ofNullable(mapper.apply(value));}
}
9. flatMap(Function<? super T,Optional> mapper)
如果存在值,则将提供的Optional方位映射函数应用于该值,返回该结果,否则返回空的Optional。此方法类似于map,但提供的映射器的结果已经是可选的,并且如果调用,flatMap不会不会在最后进行任何包装。
Optional<String> optional = optionalTestNew.flatMap(OptionalTest::getFlatMap);
log.info("flatMap后得到的字段:{}",optional.get());private static Optional<String> getFlatMap(Test test){return Optional.ofNullable(test).map(Test::getName);
}
源码查看:
也是进行非空判断,然后和map不同的是不执行ofNullable方法
public<U> Optional<U> flatMap(Function<? super T, Optional<U>> mapper) {Objects.requireNonNull(mapper);if (!isPresent())return empty();else {return Objects.requireNonNull(mapper.apply(value));}
}
10. orElse(T other)
如果有值则将其返回,否则返回指定的其它值。
如果你是一个对象,orElse()
也要是相同对象!
String message = null;
String messageNew = "关注公众号:小王博客基地";String nullString = Optional.ofNullable(message).orElse("这是一个空字符串!");
log.info("这是空字符串打印的:{}",nullString);
String string = Optional.ofNullable(messageNew).orElse("=====这是一个空字符串!");
log.info("这是字符串打印的:{}",string);
源码查看:
简单的为空返回自己定义的,不为空直接返回!
public T orElse(T other) {return value != null ? value : other;
}
11. orElseGet(Supplier<? extends T> other)
返回值(如果存在),否则调用other并返回该调用的结果。
区别:
orElse方法将传入的参数作为默认值,orElseGet方法可以接受Supplier接口的实现用来生成默认值
如果没有复杂操作,Idea也会提醒我们不要使用这个,使用orElse即可!
String message = null;
String messageNew = "关注公众号:小王博客基地";
String orElseGet = Optional.ofNullable(message).orElseGet(() -> "这还是一个空的字符串");
log.info("orElseGet调用:这是空字符串打印的:{}",orElseGet);
String orElseGetString = Optional.ofNullable(messageNew).orElseGet(() -> "这还是一个空的字符串");
log.info("orElseGet调用:这是字符串打印的:{}",orElseGetString);
源码查看:
和orElse一样,只不过为空调用lambda执行!
public T orElseGet(Supplier<? extends T> other) {return value != null ? value : other.get();
}
12. orElseThrow(Supplier<? extends X> exceptionSupplier)
返回包含的值(如果存在),否则抛出由提供的供应商创建的异常。
String message = null;
String messageNew = "关注公众号:小王博客基地";
Optional.ofNullable(messageNew).orElseThrow(() -> new RuntimeException("为空了,还不看看!"));
Optional.ofNullable(message).orElseThrow(() -> new RuntimeException("为空了,还不看看!"));
我们可以自定义异常,然后来引用!
源码查看:
为空则走自己写的异常!
public <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier) throws X {if (value != null) {return value;} else {throw exceptionSupplier.get();}
}
13. 例子汇总
/*** @author wangzhenjun* @date 2023/2/27 10:22*/
@Slf4j
public class OptionalTest {public static void main(String[] args) {Optional<Object> empty = Optional.empty();log.info("empty值:{}",empty);Test testNew = new Test();Test test = null;Optional<Test> optionalNew = Optional.of(testNew);log.info(" optional对象:{}",optionalNew);
// Optional<Test> optional = Optional.of(test);Optional<Test> optionalTest = Optional.ofNullable(test);log.info(" optional对象中的ofNullable方法返回值:{}",optionalTest);Optional<Test> optionalTestNew = Optional.ofNullable(testNew);log.info(" optional对象中的ofNullable方法new返回值:{}",optionalTestNew);Test test2 = optionalTestNew.get();log.info("原来有值的:经过Optional包装后get后得到原来的值:{}",test2);// Test test1 = optionalTest.get();// log.info("原来没有值的:经过Optional包装后get后得到原来的值:{}",test1);boolean present = optionalTestNew.isPresent();log.info("optionalTestNew调用是否为空:{}",present);boolean present1 = optionalTest.isPresent();log.info("optionalTest调用是否为空:{}",present1);optionalTest.ifPresent(new Consumer<Test>() {@Overridepublic void accept(Test test) {log.info("我是调用ifPresent执行后的打印=====");}});optionalTestNew.ifPresent(testInner -> log.info("我是调用ifPresent执行后的打印"));testNew.setName("萧炎");testNew.setAge(33);Optional<Test> optionalTest1 = optionalTestNew.filter(test1 -> test1.getAge() > 30);log.info("过滤后的结果:{}",optionalTest1.get());Optional<String> stringOptional = optionalTestNew.map(Test::getName);log.info("map后获得字段值:{}",stringOptional.get());Optional<String> optional = optionalTestNew.flatMap(OptionalTest::getFlatMap);log.info("flatMap后得到的字段:{}",optional.get());String message = null;String messageNew = "关注公众号:小王博客基地";String nullString = Optional.ofNullable(message).orElse("这是一个空字符串!");log.info("这是空字符串打印的:{}",nullString);String string = Optional.ofNullable(messageNew).orElse("=====这是一个空字符串!");log.info("这是字符串打印的:{}",string);String orElseGet = Optional.ofNullable(message).orElseGet(() -> "这还是一个空的字符串");log.info("orElseGet调用:这是空字符串打印的:{}",orElseGet);String orElseGetString = Optional.ofNullable(messageNew).orElseGet(() -> "这还是一个空的字符串");log.info("orElseGet调用:这是字符串打印的:{}",orElseGetString);Optional.ofNullable(messageNew).orElseThrow(() -> new RuntimeException("为空了,还不看看!"));Optional.ofNullable(message).orElseThrow(() -> new RuntimeException("为空了,还不看看!"));}private static Optional<String> getFlatMap(Test test){return Optional.ofNullable(test).map(Test::getName);}}
四、总结
这里就不在演示实战了,基本上组合使用:
Optional.ofNullable(需要判断的对象).ifPresent(具体操作)
其实和if相比就是显得优雅一些,主要是防止某处没考虑到,忘记if判断,那么后续可能会导致空指针,如果使用Optional的话,那么这个问题能够得到避免。
就像多使用设计模式一样,让自己的代码更加健壮优雅,还是要多使用一些的!当然不能过渡使用!!
对你有帮助,还请不要吝啬你的发财小手点点关注哈!、
写作不易,大家给点支持,你的支持是我写作的动力哈!
关注小编的微信公众号,一起交流学习!文章首发看哦!
相关文章:

告别空指针让代码变优雅,Optional使用图文例子源码解读
一、前言 我们在开发中最常见的异常就是NullPointerException,防不胜防啊,相信大家肯定被坑过! 这种基本出现在获取数据库信息中、三方接口,获取的对象为空,再去get出现! 解决方案当然简单,只…...

【C++】哈希——unordered系列容器|哈希冲突|闭散列|开散列
文章目录一、unordered系列关联式容器二、哈希概念三、哈希冲突四、哈希函数五、解决哈希冲突1.闭散列——开放定址法2.代码实现3.开散列——开链法4.代码实现六、结语一、unordered系列关联式容器 在C98中,STL提供了底层为红黑树结构的一系列关联式容器,…...
mysql-面试
锁: mysql的锁分为全局锁、表锁、行锁、间隙锁 全局锁:Flush tables with read lock 可以全局设计库为只读 表锁:一种是表锁,一种是元数据锁(meta data lock,MDL) lock tables t1 read,t2 wi…...

【夏虫语冰】Win10局域网下两台电脑无法ping通: 无法访问目标主机
文章目录1、简介2、修改高级共享设置3、启用防火墙规则4、局域网内的其他主机访问NAT模式下的虚拟机4.1 虚拟机网络设置4.2 访问测试4.2.1 http测试4.2.2 curl测试4.2.3 telnet测试4.2.4 端口占用测试5、其他结语1、简介 ping 192.168.31.134ping主机ip时,访问无法…...

大数据框架之Hadoop:MapReduce(三)MapReduce框架原理——Join多种应用
3.7.1Reduce Join 1、工作原理 Map端的主要工作:为来自不同表或文件的key/value对,打标签以区别不同来源的记录。然后用连接字段作为key,其余部分和新加的标志作为value,最后进行输出。 Reduce端的主要工作:在Reduc…...

SSRF漏洞原理、危害以及防御与修复
一、SSRF漏洞原理漏洞概述SSRF(Server-side Request Forge,服务端请求伪造)是一种由攻击者构造形成由服务端发起请求的安全漏洞。一般情况下,SSRF攻击的目标是从外网无法访问的内部系统。正是因为它是由服务端发起的,所…...

CV学习笔记-ResNet
ResNet 文章目录ResNet1. ResNet概述1.1 常见卷积神经网络1.2 ResNet提出背景2. ResNet网络结构2.1 Residual net2.2 残差神经单元2.3 Shortcut2.4 ResNet50网络结构3. 代码实现3.1 Identity Block3.2 Conv Block3.3 ResNet网络定义3.4 整体代码测试1. ResNet概述 1.1 常见卷积…...

百亿数据,毫秒级返回查询优化
近年来公司业务迅猛发展,数据量爆炸式增长,随之而来的的是海量数据查询等带来的挑战,我们需要数据量在十亿,甚至百亿级别的规模时依然能以秒级甚至毫秒级的速度返回,这样的话显然离不开搜索引擎的帮助,在搜…...

cpp之STL
STL原理 STL ⼀共提供六⼤组件,包括容器,算法,迭代器,仿函数,适配器和空间配置器,彼此可以组合套⽤。容器通过配置器取得数据存储空间,算法通过迭代器存取容器内容,仿函数可以协助算…...
基于Spring Boot开发的资产管理系统
文章目录 项目介绍主要功能截图:登录首页信息软件管理服务器管理网络设备固定资产明细硬件管理部分代码展示设计总结项目获取方式🍅 作者主页:Java韩立 🍅 简介:Java领域优质创作者🏆、 简历模板、学习资料、面试题库【关注我,都给你】 🍅文末获取源码联系🍅 项目…...
Markdown总结
文字的着重标记与段落的层次划分 Tab键可以缩进列表; shift Tab:取消缩进列表 加粗(****)、斜体(**)高亮:xxx$$:特殊标记删除:~~xxx~~多级标题:######无序列…...

字节跳动软件测试岗4轮面经(已拿34K+ offer)...
没有绝对的天才,只有持续不断的付出。对于我们每一个平凡人来说,改变命运只能依靠努力幸运,但如果你不够幸运,那就只能拉高努力的占比。 2021年10月,我有幸成为了字节跳动的一名测试工程师,从外包辞职了历…...
docker - 搭建redis集群和Etcd
概述 由于业务需要,需要把之前的分布式架构调整成微服务,把老项目迁移到k8s的服务中,再开始编码之前,需要再本地环境里做相应的准备工作,使用docker搭建redis集群,Etcd主要是注册本地的rpc服务。 Liunx O…...

Java程序开发中如何使用lntelliJ IDEA?
完成了IDEA的安装与启动,下面使用IDEA创建一个Java程序,实现在控制台上打印HelloWorld!的功能,具体步骤如下。 1.创建Java项目 进入New Project界面后,单击New Project选项按钮创建新项目,弹出New Project对话框&…...

【Linux】理解进程地址空间
🍎作者:阿润菜菜 📖专栏:Linux系统编程 我们在学习C语言的时候,都学过内存区域的划分如栈、堆、代码区、数据区这些。但我们其实并不真正理解内存 — 我们之前一直说的内存是物理上的内存吗? 前言 我们…...

Unity脚本 --- 常用API(类)--- GameObject类 和
第一部分 --- GameObject类 1.在Hierarchy 层级面板中添加游戏物体其实就相当于在场景中添加游戏物体 2.每一个场景都有一个自己的Hierarchy层级面板,用来管理场景中的所有游戏物体 3.是的,我们可以创建多个场景 1.首先上面这两个变量都是布尔变量&am…...

HTML标签——表格标签
HTML标签——表格标签 目录HTML标签——表格标签一、表格标题和表头单元格标签场景:注意点:案例实操小结二、表格的结构标签场景:注意点:案例实操:三、合并单元格思路场景:代码实现一、表格标题和表头单元格…...

Telerik JustMock 2023 R1 Crack
Telerik JustMock 2023 R1 Crack 制作单元测试的最快、最灵活和模拟选项。 Telerik JustLock也很简单,可以使用一个模拟工具来帮助您更快地生成更好的单元测试。JustLock使您更容易创建对象并建立对依赖关系的期望,例如,互联网服务需求、数据…...

筑基八层 —— 问题思考分析并解决
目录 零:移步 一.修炼必备 二.问题思考(先思考) 三.问题解答 零:移步 CSDN由于我的排版不怎么好看,我的有道云笔记相当的美观,请移步有道云笔记 一.修炼必备 1.入门必备:VS2019社区版&#x…...

【面试题】当面试官问 Vue2与Vue3的区别,你该怎么回答?
大厂面试题分享 面试题库后端面试题库 (面试必备) 推荐:★★★★★地址:前端面试题库被问到 《vue2 与 vue3 的区别》应该怎么回答Vue 内部根据功能可以被分为三个大的模块:响应性 reactivite、运行时 runtime、编辑器…...
conda相比python好处
Conda 作为 Python 的环境和包管理工具,相比原生 Python 生态(如 pip 虚拟环境)有许多独特优势,尤其在多项目管理、依赖处理和跨平台兼容性等方面表现更优。以下是 Conda 的核心好处: 一、一站式环境管理:…...

基于ASP.NET+ SQL Server实现(Web)医院信息管理系统
医院信息管理系统 1. 课程设计内容 在 visual studio 2017 平台上,开发一个“医院信息管理系统”Web 程序。 2. 课程设计目的 综合运用 c#.net 知识,在 vs 2017 平台上,进行 ASP.NET 应用程序和简易网站的开发;初步熟悉开发一…...

PPT|230页| 制造集团企业供应链端到端的数字化解决方案:从需求到结算的全链路业务闭环构建
制造业采购供应链管理是企业运营的核心环节,供应链协同管理在供应链上下游企业之间建立紧密的合作关系,通过信息共享、资源整合、业务协同等方式,实现供应链的全面管理和优化,提高供应链的效率和透明度,降低供应链的成…...

CMake基础:构建流程详解
目录 1.CMake构建过程的基本流程 2.CMake构建的具体步骤 2.1.创建构建目录 2.2.使用 CMake 生成构建文件 2.3.编译和构建 2.4.清理构建文件 2.5.重新配置和构建 3.跨平台构建示例 4.工具链与交叉编译 5.CMake构建后的项目结构解析 5.1.CMake构建后的目录结构 5.2.构…...

NLP学习路线图(二十三):长短期记忆网络(LSTM)
在自然语言处理(NLP)领域,我们时刻面临着处理序列数据的核心挑战。无论是理解句子的结构、分析文本的情感,还是实现语言的翻译,都需要模型能够捕捉词语之间依时序产生的复杂依赖关系。传统的神经网络结构在处理这种序列依赖时显得力不从心,而循环神经网络(RNN) 曾被视为…...

成都鼎讯硬核科技!雷达目标与干扰模拟器,以卓越性能制胜电磁频谱战
在现代战争中,电磁频谱已成为继陆、海、空、天之后的 “第五维战场”,雷达作为电磁频谱领域的关键装备,其干扰与抗干扰能力的较量,直接影响着战争的胜负走向。由成都鼎讯科技匠心打造的雷达目标与干扰模拟器,凭借数字射…...

多模态大语言模型arxiv论文略读(108)
CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文标题:CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文作者:Sayna Ebrahimi, Sercan O. Arik, Tejas Nama, Tomas Pfister ➡️ 研究机构: Google Cloud AI Re…...

初探Service服务发现机制
1.Service简介 Service是将运行在一组Pod上的应用程序发布为网络服务的抽象方法。 主要功能:服务发现和负载均衡。 Service类型的包括ClusterIP类型、NodePort类型、LoadBalancer类型、ExternalName类型 2.Endpoints简介 Endpoints是一种Kubernetes资源…...

RSS 2025|从说明书学习复杂机器人操作任务:NUS邵林团队提出全新机器人装配技能学习框架Manual2Skill
视觉语言模型(Vision-Language Models, VLMs),为真实环境中的机器人操作任务提供了极具潜力的解决方案。 尽管 VLMs 取得了显著进展,机器人仍难以胜任复杂的长时程任务(如家具装配),主要受限于人…...
CRMEB 中 PHP 短信扩展开发:涵盖一号通、阿里云、腾讯云、创蓝
目前已有一号通短信、阿里云短信、腾讯云短信扩展 扩展入口文件 文件目录 crmeb\services\sms\Sms.php 默认驱动类型为:一号通 namespace crmeb\services\sms;use crmeb\basic\BaseManager; use crmeb\services\AccessTokenServeService; use crmeb\services\sms\…...