告别丑陋判空,一个Optional类就能搞定

news2024/12/25 1:47:58

认识Optional

Opitonal类就是Java提供的为了解决大家平时判断对象是否为空用,通常会用 null!=obj 这样的方式存在的判断,从而令人头疼导致空指针异常,同Optional的存在可以让代码更加简单,可读性跟高,代码写起来更高效

Student student = new Student();
if (null == student){
    return "student为null";
}
return student;
Student student = new Student();
return Optional.ofNullable(student).orElse("student为null");

测试展示类Student 代码(如果有朋友不明白可以看一下这个):

@Data
@AllArgsConstructor
@NoArgsConstructor
public class Student {
    private String name;
    private Integer age;
}

Optional对象创建

首先我们先打开Optional的内部,去一探究竟 先把几个创建Optional对象的方法提取出来

public final class Optional<T> {
   private static final Optional<?> EMPTY = new Optional<>();
   private final T value;
   // 我们可以看到两个构造方格都是private 私有的
   // 说明 我们没办法在外面去new出来Optional对象
   private Optional() {
        this.value = null;
    }
   private Optional(T value) {
        this.value = Objects.requireNonNull(value);
    }
    // 这个静态方法大致 是创建出一个包装值为空的一个对象因为没有任何参数赋值
   public static<T> Optional<T> empty() {
        @SuppressWarnings("unchecked")
        Optional<T> t = (Optional<T>) EMPTY;
        return t;
    }
    // 这个静态方法大致 是创建出一个包装值非空的一个对象 因为做了赋值
   public static <T> Optional<T> of(T value) {
        return new Optional<>(value);
    }
    // 这个静态方法大致是 如果参数value为空,则创建空对象,如果不为空,则创建有参对象
   public static <T> Optional<T> ofNullable(T value) {
        return value == null ? empty() : of(value);
    }
 }

再做一个简单的实例展示 与上面对应

// 1、创建一个包装对象值为空的Optional对象
Optional<String> optEmpty = Optional.empty();
// 2、创建包装对象值非空的Optional对象
Optional<String> optOf = Optional.of("optional");
// 3、创建包装对象值允许为空也可以不为空的Optional对象
Optional<String> optOfNullable1 = Optional.ofNullable(null);
Optional<String> optOfNullable2 = Optional.ofNullable("optional");

Optional.get()方法(返回对象的值)

get()方法是返回一个option的实例值

源码:

public T get() {
    if (value == null) {
        throw new NoSuchElementException("No value present");
    }
    return value;
}

也就是如果value不为空则做返回,如果为空则抛出异常 “No value present” 简单实例展示

Student student = new Student();
student.setAge(18);
System.out.println(Optional.ofNullable(student).get());

Optional.isPresent()方法(判读是否为空)

isPresent()方法就是会返回一个boolean类型值,如果对象不为空则为真,如果为空则false

源码:

public boolean isPresent() {
    return value != null;
}

简单的实例展示:

Student student = new Student();
student.setAge(18);
if (Optional.ofNullable(student).isPresent()){
    System.out.println("不为空");
}else {
    System.out.println("为空");
}

Optional.ifPresent()方法(判读是否为空并返回函数)

这个意思是如果对象非空,则运行函数体

源码:

public void ifPresent(Consumer<? super T> consumer) {
    //如果value不为空,则运行accept方法体
    if (value != null)
        consumer.accept(value);
}

看实例:

Student student = new Student();
student.setAge(18);
Optional.ofNullable(student).ifPresent(s -> System.out.println("年龄:" + s.getAge()));

如果对象不为空,则会打印这个年龄,因为内部已经做了NPE(非空判断),所以就不用担心空指针异常了

Optional.filter()方法(过滤对象)

filter()方法大致意思是,接受一个对象,然后对他进行条件过滤,如果条件符合则返回Optional对象本身,如果不符合则返回空Optional

源码:

public Optional<T> filter(Predicate<? super T> predicate) {
    Objects.requireNonNull(predicate);
    //如果为空直接返回this
    if (!isPresent())
        return this;
    else
        //判断返回本身还是空Optional
        return predicate.test(value) ? this : empty();
}

简单实例:

Student student = new Student();
student.setAge(18);
Optional.ofNullable(student).filter(s -> s.getAge()>10);

Optional.flatMap()方法(Optional对象进行二次包装)

map()方法将对应Optional< Funcation >函数式接口中的对象,进行二次运算,封装成新的对象然后返回在Optional中

源码:

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));
    }
}

实例:

Student student = new Student();
student.setAge(18);
Optional<Object> optName = Optional.ofNullable(student).map(s -> Optional.ofNullable(s.getName()).orElse("name为空"));

Optional.orElse()方法(为空返回对象)

常用方法之一,这个方法意思是如果包装对象为空的话,就执行orElse方法里的value,如果非空,则返回写入对象

源码:

public T orElse(T other) {
    //如果非空,返回value,如果为空,返回other
    return value != null ? value : other;
}

实例:

Student student = new Student();
student.setAge(18);
Optional.ofNullable(student).orElse(new Student("小明", 12));

Optional.orElseGet()方法(为空返回Supplier对象)

这个与orElse很相似,入参不一样,入参为Supplier对象,为空返回传入对象的.get()方法,如果非空则返回当前对象。推荐:Java面试题

源码:

public T orElseGet(Supplier<? extends T> other) {
    return value != null ? value : other.get();
}

实例:

Optional<Supplier<Student>> sup=Optional.ofNullable(Student::new);
//调用get()方法,此时才会调用对象的构造方法,即获得到真正对象
Optional.ofNullable(student).orElseGet(sup.get());

Suppiler是一个接口,是类似Spring的懒加载,声明之后并不会占用内存,只有执行了get()方法之后,才会调用构造方法创建出对象 创建对象的语法的话就是Supplier supStudent= Student::new; 需要使用时supStudent.get()即可

Optional.orElseThrow()方法(为空返回异常)

方法作用的话就是如果为空,就抛出你定义的异常,如果不为空返回当前对象,在实战中所有异常肯定是要处理好的,为了代码的可读性

源码:

public <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier) throws X {
    if (value != null) {
        return value;
    } else {
        throw exceptionSupplier.get();
    }
}

实例:

//简单的一个查询
Member member = memberService.selectByPhone(request.getPhone());
Optional.ofNullable(member).orElseThrow(() -> new ServiceException("没有查询的相关数据"));

相似方法进行对比分析

可能看到这,没用用过的话会觉得orElse()orElseGet()还有orElseThrow()很相似,map()和flatMap()好相似,不用着急,都是从这一步过来的,我再给大家总结一下不同方法的异同点 orElse()orElseGet()orElseThrow()的异同点

方法效果类似,如果对象不为空,则返回对象,如果为空,则返回方法体中的对应参数,所以可以看出这三个方法体中参数是不一样的 orElse(T 对象) orElseGet(Supplier < T >对象) orElseThrow(异常)

map()和orElseGet的异同点

方法效果类似,对方法参数进行二次包装,并返回,入参不同 map(function函数) flatmap(Optional< function >函数)

具体要怎么用,要根据业务场景以及代码规范来定义,下面可以简单看一下我在实战中怎用使用神奇的Optional。推荐:Java面试题

实战场景

场景1:在service层中 查询一个对象,返回之后判断是否为空并做处理

//查询一个对象
Member member = memberService.selectByIdNo(request.getCertificateNo());
//使用ofNullable加orElseThrow做判断和操作
Optional.ofNullable(member).orElseThrow(() -> new ServiceException("没有查询的相关数据"));

场景2:我们可以在dao接口层中定义返回值时就加上Optional

public interface LocationRepository extends JpaRepository<Location, String> {
Optional<Location> findLocationById(String id);
}

然在是Service中

public TerminalVO findById(String id) {
    //这个方法在dao层也是用了Optional包装了
    Optional<Terminal> terminalOptional = terminalRepository.findById(id);
    //直接使用isPresent()判断是否为空
    if (terminalOptional.isPresent()) {
    //使用get()方法获取对象值
        Terminal terminal = terminalOptional.get();
        //在实战中,我们已经免去了用set去赋值的繁琐,直接用BeanCopy去赋值
        TerminalVO terminalVO = BeanCopyUtils.copyBean(terminal, TerminalVO.class);
        //调用dao层方法返回包装后的对象
        Optional<Location> location = locationRepository.findLocationById(terminal.getLocationId());
        if (location.isPresent()) {
            terminalVO.setFullName(location.get().getFullName());
        }
        return terminalVO;
    }
    //不要忘记抛出异常
    throw new ServiceException("该终端不存在");
}

实战场景还有很多,包括return时可以判断是否返回当前值还是跳转到另一个方法体中,其它的还有很多。

Optional使用注意事项

Optional真么好用,真的可以完全替代if判断吗?我想这肯定是大家使用完之后Optional之后可能会产生的想法,答案是否定的 举一个最简单的栗子:

例子:如果我只想判断对象的某一个变量是否为空并且做出判断呢?

Person person=new Person();
person.setName("");
persion.setAge(2);
//普通判断
if(StringUtils.isNotBlank(person.getName())){
   //名称不为空执行代码块
}
//使用Optional做判断
Optional.ofNullable(person).map(p -> p.getName()).orElse("name为空");

我觉得这个例子就能很好的说明这个问题,只是一个很简单判断,如果用了Optional我们还需要考虑包装值,考虑代码书写,考虑方法调用,虽然只有一行,但是可读性并不好,如果别的程序员去读,我觉得肯定没有if看的明显。

Jdk 9对Optional优化

首先增加了三个方法: or()ifPresentOrElse()stream()。or() 与orElse等方法相似,如果对象不为空返回对象,如果为空则返回or()方法中预设的值。

ifPresentOrElse() 方法有两个参数:一个 Consumer 和一个 Runnable。如果对象不为空,会执行 Consumer 的动作,否则运行 Runnable。相比ifPresent()多了OrElse判断。stream()将Optional转换成stream,如果有值就返回包含值的stream,如果没值,就返回空的stream。

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

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

相关文章

剑指 Offer 16. 数值的整数次方

题目 实现 pow(x, n) &#xff0c;即计算 x 的 n 次幂函数&#xff08;即&#xff0c;xn&#xff09;。不得使用库函数&#xff0c;同时不需要考虑大数问题。 思路 快速幂算法 求 x^n最简单的方法是通过循环将 n 个 x 乘起来&#xff0c;依次求 x1,x2,...,x^n&#xff0c;…

【微服务】Nacos 注册中心服务数据模型

目录 一、前言 二、服务&#xff08;Service&#xff09;和服务实例&#xff08;Instance&#xff09; 1、定义服务 2、服务元数据 3、定义实例 4、实例元数据 5、持久化属性 三、集群 1、定义集群 四、生命周期 1、服务的⽣命周期 2、实例的⽣命周期 3、集群的⽣命…

使用Comsol完成固体火箭装药的燃面推移

简介 为了计算固体火箭发动机的内弹道曲线&#xff0c;必须知道燃面面积随烧去肉厚的变化关系。目前主要有实体造型法、动网格法、水平集法等。笔者给出一种直接使用Comsol有限元软件完成燃面推移的方法&#xff0c;使用简便快捷&#xff0c;无需编写代码。 理论部分可参考论…

API风险

©网络研究院 从表面上看&#xff0c;API帮助企业连接应用程序并相互共享数据。这为客户和用户创造了更简单、更无缝的体验。 如果你曾经使用你的谷歌账户登录过多个网站或应用程序&#xff0c;那么你很有可能使用的是谷歌开发的API。 像这样的API在后台工作&#xff0c…

极狐gitlab-runner的安装

1&#xff09;下载系统对应的二进制文件 # Linux x86-64 sudo curl -L --output /usr/local/bin/gitlab-runner https://gitlab-runner-downloads.s3.amazonaws.com/latest/binaries/gitlab-runner-linux-amd64 2&#xff09;赋予执行权限 chmod x /usr/local/bin/gitlab-runne…

Faster RCNN网络源码解读(Ⅱ) --- Faster RCNN源码使用

目录 一、源码链接 二、环境配置 三、文件结构 四、预训练权重下载地址 五、训练集 六、训练方法及注意事项 七、大概看一下训练过程&#xff08;train_mobilenetv2.py&#xff09; 一、源码链接 Faster R-CNN源码链接https://pan.baidu.com/s/1SQjyLXD47H11ke05OXY…

基于javaweb的学院社团管理系统(idea+servlet+jsp)

一、系统简介 本项目采用idea工具开发&#xff0c;jspservletjquery技术编写&#xff0c;数据库采用的是mysql&#xff0c;navicat开发工具。 系统一共分为3个角色分别是&#xff1a;管理员&#xff0c;学生&#xff0c;社长 获取方式&#xff1a;基于javaweb的学院社团管理系…

VisualStudio2015安装不上的解决方法_选择对应msi_依然报错继续选择---VisualStudio2015工作笔记001

这个visualstudio网上竟然有卖的,真的是太~咱啥也不说了~ 好了说解决办法吧; 1.这里完整版大小挺大的6G多,但是这里我们找的有可能是只有3.8G左右的,这个版本的就是有文件损坏...但是这个版本是可以装上的,也不耽误用的,我亲自测试了. 2.因为很久没写vb.net程序了,想写个小程…

Docker中安装宝塔

1、docker拉取ubuntu系统 docker pull ubuntudocker pull ubuntu 2、运行容器 docker run -i -t -d --name bt -p 2000:20 -p 2100:21 -p 8000:80 -p 4430:443 -p 8880:888 -p 8888:8888 --privilegedtrue -v /d/baota:/www/wwwroot ubuntu -v后的/d/baota代表本地D盘的bao…

【LeetCode】二叉树中的最大路径和 [H](递归)

124. 二叉树中的最大路径和 - 力扣&#xff08;LeetCode&#xff09; 一、题目 路径 被定义为一条从树中任意节点出发&#xff0c;沿父节点-子节点连接&#xff0c;达到任意节点的序列。同一个节点在一条路径序列中 至多出现一次 。该路径 至少包含一个 节点&#xff0c;且不一…

擎创技术流 | ClickHouse实用工具—ckman教程(8)

2022年即将迎来终点&#xff0c;各大公司和网站开始出“年终使用报告”了&#xff0c;而关于ClickHouse系列分享也来到了尾期。本系列最初一共规划了10期内容&#xff0c;一转眼就只剩最后2期了&#xff0c;初衷就是单纯的技术分享&#xff0c;如果你能从中得到启发&#xff0c…

圣诞节来临,TikTok上圣诞相关产品销量突破5万单丨超店有数

12月25日&#xff0c;圣诞节正式来临。TikTok卖家和达人如何把握住这个一年一度的「西方春节」&#xff0c;如何实现精准选品&#xff0c;快速爆单。凡事预则立&#xff0c;不预则废。快人一步&#xff0c;掌握趋势&#xff0c;抢占先机。 从谷歌趋势观察&#xff0c;一周内「…

图说Netty服务端启动过程

我们知道Netty是一个基于JDK的nio实现的网络编程框架&#xff0c;那Netty的服务端是怎么启动的呢&#xff0c;包括他是何时register 的&#xff0c;何时 bind 端口的&#xff0c;以及何时开始读取网络中的数据的&#xff1f; 让我们带着这个疑问&#xff0c;通过一个官方的例子…

webRTC 实现人脸识别

首先我们需要先了解一下什么是webRTC 他能做什么 webRTC主要是帮我们处理多媒体应用&#xff0c;如音视频通话&#xff0c;屏幕共享都可以实现&#xff0c;主要基于浏览器API调用&#xff0c;其底层浏览器会调用native C 等一些库帮我们实现的&#xff0c;而我们在应用层掉API …

欧科云链荣获人民网匠心技术奖,科技创新共造企业发展“强引擎”

什么是匠心精神&#xff1f; 是简单的事情重复做 重复的事情用心做 是对自己热爱的事物 不断坚持、不断钻研的精神 昨日&#xff0c;由人民网主办的“2022人民财经高峰论坛”成功举办&#xff0c;论坛还公布了“第十九届人民匠心奖”获奖名单。欧科云链携旗下“链上天眼”产品荣…

第三十七章 数论——博弈论

第三十七章 数论——博弈论一、Nim游戏1、题目2、结论3、结论验证4、代码二、台阶——Nim游戏1、问题2、思路2、代码三、集合——Nim游戏1、问题2、思路—SG()函数2、代码实现&#xff08;记忆化搜索&#xff09;一、Nim游戏 1、题目 2、结论 这里直接说结论&#xff1a; 假设…

vue导入私有组件和注册全局组件

目录先下载并配置插件导入私有组件注册全局组件先下载并配置插件 导入的时候需要路径,有个符号,但不能提示路径,需要手打路径,会发现很麻烦,这时候可以通过vscode插件来解决 vscode搜索Path Autocomplete 配置插件,点击插件设置—扩展设置,点开任意一个setting.json中编辑,打开…

Yield Guild Games 和 Axie Infinity:迄今为止的旅程

2022 年&#xff0c;菲律宾被认为是全球应用 web3 的中心&#xff0c;在 Chainalysis 的全球加密货币应用指数中排名第二&#xff0c;在拥有最多 MetaMask 用户的国家中排名第三。虽然该国作为 Coins.ph 和 BloomX 等开创性数字资产交易所以及对加密货币友好的 UnionBank 的所在…

Flutter生命周期

一、组件生命周期 flutter组件只有两种&#xff1a;有状态和无状态组件。由于无状态组件效率高&#xff0c;如果不涉及到组件内部的数据存储&#xff0c;尽量多的使用无状态组件 1、StatelessWidget build&#xff1a;组件渲染 调用次数&#xff1a;1次 StatelessWidget是无状…

MYSQL索引和sql优化

基本 索引是帮助数据高效查询的有序数据结构&#xff0c;没有索引进行查询就会进行全表扫描myisam中的.MYI和innodb中的.idb都是存放索引的文件。索引提高查询效率的同时&#xff0c;也降低了更新表的数据&#xff0c;因为数据库中删改查会维护索引的结构。一般提到的索引就是…