Java8 Optional类使用总结

前言

相信不少小伙伴已经被java的NPE(NullPointer Exception)所谓的空指针异常搞的头昏脑涨, 有大佬说过“防止 NPE,是程序员的基本修养。”但是修养归修养,也是我们程序员最头疼的问题之一,那么我们今天就要尽可能的利用Java8的新特性 Optional来尽量简化代码同时高效处理NPE(NullPointer Exception 空指针异常)。

认识Optional并使用

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

//常规判断:  
//对象 人  
//属性有 name,age  
Person person=new Person();  
if (null==person){  
    return "person为null";  
}  
return person;
//使用Optional:  
//对象 人  
//属性有 name,age  
Person person=new Person();  
return Optional.ofNullable(person).orElse("person为null");

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

public class Person {  
    private String name;  
    private Integer age;  
  
    public Person(String name, Integer age) {  
        this.name = name;  
        this.age = age;  
    }  
  
    public Person() {  
    }  
  
    public String getName() {  
        return name;  
    }  
  
    public void setName(String name) {  
        this.name = name;  
    }  
  
    public Integer getAge() {  
        return age;  
    }  
  
    public void setAge(Integer age) {  
        this.age = age;  
    }  
}

下面,我们就高效的学习一下神奇的Optional类!

2.1 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对象的内部方法大致分析完毕 接下来也正式的进入Optional的学习与使用中

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

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

源码:

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

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

Person person=new Person();  
person.setAge(2);  
Optional.ofNullable(person).get();

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

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

源码:

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

简单的实例展示:

Person person=new Person();  
person.setAge(2);  
if (Optional.ofNullable(person).isPresent()){  
    //写不为空的逻辑  
    System.out.println("不为空");  
}else{  
    //写为空的逻辑  
    System.out.println("为空");  
}

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

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

源码:

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

看实例:

Person person=new Person();  
person.setAge(2);  
Optional.ofNullable(person).ifPresent(p -> System.out.println("年龄"+p.getAge()));

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

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

简单实例:

Person person=new Person();  
person.setAge(2);  
Optional.ofNullable(person).filter(p -> p.getAge()>50);

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

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

源码:

public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {  
    Objects.requireNonNull(mapper);  
    //如果为空返回自己  
    if (!isPresent())  
        return empty();  
    else {  
        //否则返回用方法修饰过的Optional  
        return Optional.ofNullable(mapper.apply(value));  
    }  
}

实例展示:

Person person1=new Person();  
person.setAge(2);  
String optName = Optional.ofNullable(person).map(p -> person.getName()).orElse("name为空");

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

实例:

Person person=new Person();  
person.setAge(2);  
Optional<Object> optName = Optional.ofNullable(person).map(p -> Optional.ofNullable(p.getName()).orElse("name为空"));

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

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

源码:

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

实例:

Person person1=new Person();  
person.setAge(2);  
Optional.ofNullable(person).orElse(new Person("小明", 2));

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

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

源码:

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

实例:

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

说真的对于Supplier对象我也懵逼了一下,去网上简单查阅才得知 Supplier也是创建对象的一种方式,简单来说,Suppiler是一个接口,是类似Spring的懒加载,声明之后并不会占用内存,只有执行了get()方法之后,才会调用构造方法创建出对象 创建对象的语法的话就是Supplier supPerson= Person::new; 需要使用时supPerson.get()即可

2.10 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("没有查询的相关数据"));

2.11 相似方法进行对比分析

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

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

map()和orElseGet的异同点

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

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

实战场景再现

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

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

场景2:我们可以在dao接口层中定义返回值时就加上Optional 例如:我使用的是jpa,其他也同理

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之后可能会产生的想法,答案是否定的 举一个最简单的栗子:

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

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。

因为这个Jdk 9的Optional具体我没有测试,同时也发现有蛮好的文章已经也能让大家明白Jdk 9的option的优化,我就不深入去说了。

来源:juejin.cn/post/6844904154075234318

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【Linux系列】 离线安装vnc 可视化桌面

离线安装vnc 可视化桌面 缘下载安装vnc初始化链接 缘 项目需要下载 下载地址&#xff1a; http://mirror.centos.org/centos/7/updates/x86_64/Packages/tigervnc-license-1.8.0-31.el7_9.noarch.rpm http://mirror.centos.org/centos/7/os/x86_64/Packages/libXfont2-2.0.…

Word插件开发

VSTO是Visual Studio Tools for Office的简称&#xff0c;它是Microsoft Visual Studio的一个扩展&#xff0c;用于开发基于Microsoft Office平台的应用程序。VSTO提供了一套API和工具&#xff0c;使开发人员能够利用Visual Studio IDE来开发定制的Office解决方案。 在 Visual…

DiffusionGAN ——最快的小波扩散模型应用研究

介绍 扩散模型最近出现并迅速发展&#xff0c;吸引了许多研究人员的兴趣。这些模型能从随机的噪声输入生成高质量的图像。在图像生成任务中&#xff0c;它们的表现尤其优于最先进的生成模型&#xff08;GANs&#xff09;。扩散模型可以灵活地处理各种条件输入&#xff0c;从而…

knife4j swagger 使用笔记

1.接口访问的端口跟后台设置的不一致&#xff0c;接口请求无反应 处理办法 2.响应参数不显示问题 &#xff08;1&#xff09;返回的参数里面一定要有响应的参数对象&#xff0c;如下&#xff1a; &#xff08;2&#xff09;TableDataInfo 定义成泛型类 TableDataInfo package…

移动应用安全

移动应用安全 移动应用安全主要关注Android、iOS、Windows Phone等平台上移动应用软件安全状态。它涉及应用程序在其设计运行的平台上下文中的安全问题、它们使用的框架以及预期的用户集。所有主流的移动平台都提供大量可选的安全控制&#xff0c;旨在帮助软件开发人员构建安全…

浅析扩散模型与图像生成【应用篇】(十八)——ControlNet

18. Adding Conditional Control to Text-to-Image Diffusion Models 现有的文生图模型如Stable Diffusion通常需要人工输入非常准确的提示词&#xff0c;而且生成的结果还是完全随机不可控制的&#xff0c;只能通过生成多个结果&#xff0c;再从中选取最佳方案。而ControlNet的…

竞争分析:波特五力模型

波特五力模型是分析企业竞争环境的一个分析模型。 根据波特的观点&#xff0c;每家企业都受到“直接竞争对手、顾客、供应商、潜在新进公司和替代性产品”这五个“竞争作用力”的影响。 我们用波特五力模型试着分析下实体书店竞争是否激励。 直接竞争对手&#xff1a;如果直接…

料堆体积测量新方案:激光雷达

激光雷达测量料堆体积是一种高效且精确的方法。激光雷达的工作原理与雷达相似&#xff0c;通过发射激光束探测目标的位置、速度等特征量。在测量料堆体积时&#xff0c;激光雷达系统向料堆发射激光束&#xff0c;然后接收从料堆表面反射回来的信号。通过对这些反射信号的处理和…

Linux网络之DNS域名解析

一、DNS概述 1.1什么是DNS 域名解析协议&#xff0c;将域名转换成IP地址 1.2为什么要用DNS IP地址不便于记忆&#xff0c;DNS使用户可以通过易记的域名快速访问各种网络资源。 192.168.0.0—— ip地址过长而且都是数字&#xff0c;不方便记忆就出现了域名 www.baidu.com—…

记一次线上日志堆栈不打印问题排查(附:高并发系统日志打印方案可收藏)

目录 一.线上的日志堆栈不打印了二.一步一步仔细排查三.最后搞定四.聊一聊线上日志到底应该怎么打印4.1 日志打印的诉求4.2 常见的系统日志上报方案4.2.1 ELK 方案4.2.2 自定义log appender 完成应用日志采集. 4.3 日志常见框架傻傻分不清4.4 日志在高并发系统中需要注意的 tip…

神仙级Python入门教程,手把手教你从0到精通,学不会算我输!

亲爱的朋友们&#xff0c;你是否对编程充满好奇&#xff0c;却觉得它遥不可及&#xff1f; 你是否想学习一门强大的编程语言&#xff0c;却不知从何下手&#xff1f; 那么&#xff0c;这篇“神仙级”Python入门教程就是为你量身打造的&#xff01;不论你是编程小白还是有一定…

linux笔记4--shell命令1

文章目录 一. 目录1.说明2.盘符3.linux根目录(以Ubuntu为例)①说明②根目录下一些文件夹的解析/home/root/mnt/media/var/cdrom/etc/lib (/lib32--32位的&#xff0c;/lib64-64位的)/lostfound/boot/proc/bin/sbin/snap/srv/usr/opt/dev/run/tmp 二. ls命令--操作文件夹1.说明2…

探索阿里巴巴商品详情API接口:开启电商数据之旅

阿里巴巴商品详情API接口是阿里巴巴开放平台提供的一项服务&#xff0c;它允许开发者通过调用接口获取指定商品的详细信息&#xff0c;包括商品标题、价格、库存、描述、图片等。这些数据对于电商从业者来说具有极高的价值&#xff0c;可以帮助他们更好地了解市场动态&#xff…

层次分析法(AHP)计算原理解释

AHP层次分析法是一种解决多目标复杂问题的定性和定量相结合进行计算决策权重的研究方法。该方法将定量分析与定性分析结合起来&#xff0c;用决策者的经验判断各衡量目标之间能否实现的标准之间的相对重要程度&#xff0c;并合理地给出每个决策方案的每个标准的权数&#xff0c…

vue下载文件时显示进度条

1.单个下载&#xff08;开始是导出按钮 下载显示进度条&#xff09; html <el-button click.stop"exportReport(scope.row, scope.index)" v-if"!scope.row.schedule" icon"el-icon-download"size"small" type"text"styl…

Linux第十五章

&#x1f436;博主主页&#xff1a;ᰔᩚ. 一怀明月ꦿ ❤️‍&#x1f525;专栏系列&#xff1a;线性代数&#xff0c;C初学者入门训练&#xff0c;题解C&#xff0c;C的使用文章&#xff0c;「初学」C&#xff0c;linux &#x1f525;座右铭&#xff1a;“不要等到什么都没有了…

系统思考—企业辅导咨询

从2004年、2014年到2024年&#xff0c;国九条政策的发布与变迁不仅影响了行业趋势&#xff0c;更深刻地改变了企业的风险预估和策略辅导。彼得杜鲁克曾经说过&#xff1a;“必须系统地抛弃旧知识。”这不仅是企业领导者的挑战&#xff0c;也是我们每个人的难题。难点不在于我们…

GITEE 基于OAuth2的API V5版本

为了构建更好的码云生态环境&#xff0c;我们推出了基于OAuth2的API V5版本。 API V5接口使用方式以及Url都参照GitHub&#xff0c;为了各位开发者更好的兼容已经存在的第三方应用。 API 使用条款 OSCHINA 用户是资源的拥有者&#xff0c;需尊重和保护用户的权益。不能在应用…

Spring AI 抢先体验,5 分钟玩转 Java AI 应用开发

作者&#xff1a;刘军 Spring AI 是 Spring 官方社区项目&#xff0c;旨在简化 Java AI 应用程序开发&#xff0c;让 Java 开发者像使用 Spring 开发普通应用一样开发 AI 应用。 Spring Cloud Alibaba AI 以 Spring AI 为基础&#xff0c;并在此基础上提供阿里云通义系列大模…

XY_RE复现(二)

一&#xff0c;何须相思煮余年 0x55 0x8b 0xec 0x81 0xec 0xa8 0x0 0x0 0x0 0xa1 0x0 0x40 0x41 0x0 0x33 0xc5 0x89 0x45 0xfc 0x68 0x9c 0x0 0x0 0x0 0x6a 0x0 0x8d 0x85 0x60 0xff 0xff 0xff 0x50 0xe8 0x7a 0xc 0x0 0x0 0x83 0xc4…
最新文章