关于Java 中的Optional的一些事

开始

定义一个实体类User,实现get、set方法

typescript 复制代码
public class User {
    String name;
    String sex;
    public User(String name, String sex) {
        this.name = name;
        this.sex = sex;
    }
    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getSex() {
        return sex;
    }

    public void setSex(String sex) {
        this.sex = sex;
    }
}

定义一个UserRepository,用户仓库,实现仓库里面对数据库的user表进行username字段查询,假设user表里面有了"张三"这条数据,当查询到张三的时候,我们就会返回一"张三"对象,没有"张三"这条数据,就会返回null

typescript 复制代码
public class UserRepository {
    public User findUserByName(String name) {
        if (name.equals("张三")) {
            return new User("张三", "男");
        } else {
            return null;
        }
    }
}

再main函数中进行引用,实例化UserRepository对象,调用里的findUserByName方法,然后打印性别,当我们查询的"张三"的是时候,我们是假设了user表里面有一条张三,那么就会返回这个张三的数据,此时打印出的性别是"男"。

typescript 复制代码
public class Main {
    public static void main(String[] args) {
        UserRepository userRepository = new UserRepository();
        User user= userRepository.findUserByName("张三");
        System.out.println(user.getSex()); //男
    }
}

这个时候我们去查询"李四",就会出现报错;"NullPointerException" 因为表里面只有"张三"这条数据,没有李四,就会返回为null,然后,user.getSex()就会报错,都没有这个人,你还想知道他的性别。

typescript 复制代码
public class Main {
    public static void main(String[] args) {
        UserRepository userRepository = new UserRepository();
        User user= userRepository.findUserByName("李四");
        System.out.println(user.getSex());
    }
}

这个时候我们就需要加入if判断了,当有这个人,就是打印他的性别, 没有,就提示没有这个人。

csharp 复制代码
if (user != null) {
     System.out.println(user.getSex());
   } else {
      System.out.println("没有这个人");
     }

此时我么就避免了"NullPointerException"空指针异常的问题,那我们联想下,用户进行查数据的时候,他根本不知道里面有没有他想要的数据。当数据量很大的时候,都要去使用if去判断吗,这样会显的代码非常臃肿,增加了代码的阅读性,难以维护,此时Optioanl解决了我们的问题。

Optional

可以看做一个容器,里面可以包含任何东西,也可以什么都不包含, 对UserRepository的findUserByName方法进行修改,

vbnet 复制代码
public Optional<User> findUserByName(String name) {
        if (name.equals("张三")) {
            return Optional.of(new User("张三", "男"));
        } else {
            return Optional.empty();
        }
    }

ifPresent和ifPresentOrElse

再main中进行修改,使用Optional容器来包含User,ifPresent方法,如果optionalUser里面含值就会执行user -> System.out.println(user.getSex()),没有就不会执行,就会报错。

ini 复制代码
public static void main(String[] args) {
        UserRepository userRepository = new UserRepository();
        Optional<User> optionalUser= userRepository.findUserByName("李四");
        optionalUser.ifPresent(user -> System.out.println(user.getSex()));
    }

如果值为空的时候,我们想执行一些其他的操作。就需要用到ifPresentOrElse方法,有值就会执行第一个参数,没有值就会执行第二个参数。

less 复制代码
Optional<User> optionalUser= userRepository.findUserByName("李四");

optionalUser.ifPresentOrElse(
                user -> System.out.println(user.getSex()),
                () -> System.out.println("没有这个人"));
//没有这个人
    }

Optional的使用

Optional中的静态方法

of();

于创建一个包含指定值的Optional对象,使用of方法时必须要确定是有值的,

ini 复制代码
String txt = null;
Optional<String> optionalString  = Optional.of(txt);
//就会抛出NullPointerException,

ofNullable();

创建一个包含指定值的Optional对象可以有值也可以是空的,这个时候就不会出现异常了。

ini 复制代码
String txt = null;
Optional<String> optionalString  = Optional.ofNullable(txt);

empty();

创建一个空的Optional对象。

Optional中的实例方法

orElseThrow()

不带参数

ini 复制代码
String txt = null;
        Optional<String> optionalString  = Optional.ofNullable(txt);
        optionalString.orElseThrow();
//此时就会抛出一个异常:NoSuchElementException

带参数,自定义异常

vbnet 复制代码
String txt = null;
        Optional<String> optionalString  = Optional.ofNullable(txt);
        optionalString.orElseThrow(() ->
            new RuntimeException("Optional is null")
        );
// 抛出的异常为Optional is null

orElse()和orElseGet()方法

直接看代码,让你更好的理解

arduino 复制代码
public static void main(String[] args) {
        String txt = "zhangsan";
        Optional<String> optionalString  = Optional.ofNullable(txt);
        txt = optionalString.orElse(printf());
        System.out.println("txt的值是:" + txt);

        String txt1 = optionalString.orElseGet(() -> printf());
        System.out.println("txt1的值是:" + txt1);
    }

    public static String printf() {
        System.out.println("执行了printf方法");
        return "liSi";
    }
//打印结果:
//执行了printf方法
//txt的值是:zhangsan
//txt1的值是:zhangsan

此时我们可以看到: Optional有值时,orElse会执行,而orElseGet不会执行。 既然执行了orElse方法,并且我们返回的是"liSi",为什么打印变量txt值时候又是打印的"zhangSan",不是"liSi"我们可以看源码:

typescript 复制代码
public T orElse(T other) {
        return value != null ? value : other;
    }

其实里面的注释也说的很清楚了:If a value is present, returns the value, otherwise returns other. 存在这个值,就返回这个值,不存在,就返回其他。

现在我看下没有值的情况下是怎么样的:

typescript 复制代码
public static void main(String[] args) {
        String txt = null;
        Optional<String> optionalString  = Optional.ofNullable(txt);
        txt = optionalString.orElse(printf());
        System.out.println("txt的值是:" + txt);

        String txt1 = optionalString.orElseGet(() -> printf());
        System.out.println("txt1的值是:" + txt1);
    }

    public static String printf() {
        System.out.println("执行了printf方法");
        return "liSi";
    }
//打印结果:
执行了printf方法
txt的值是:liSi
执行了printf方法
txt1的值是:liSi

可以看到: Optional值为null的时候,orElse会执行,orElseGet也会执行。

我们再看下orElseGet源码

typescript 复制代码
public T orElseGet(Supplier<? extends T> supplier) {
        return value != null ? value : supplier.get();
    }

注释:If a value is present, returns the value, otherwise returns the result produced by the supplying function. 存在值,则返回该值,否则返回提供函数产生的结果。

总结

说白了就是传值就用orElse,需要调用方法获取就用orElseGet, Optional有没有值的时候都会调用orElse,而orElseGet只有Optional没有值的时候才会被调用。

相关推荐
小胖伦的夕阳粉2 分钟前
js 获取树节点上某节点的最底层叶子节点数据
开发语言·javascript·ecmascript
不见长安见晨雾11 分钟前
将Java程序打包成EXE程序
java·开发语言
@听风吟18 分钟前
力扣之182.查找重复的电子邮箱
大数据·javascript·数据库·sql·leetcode
逸狼35 分钟前
【JavaEE初阶】多线程(5 单例模式 \ 阻塞队列)
java·开发语言
唐家小妹1 小时前
【flex-grow】计算 flex弹性盒子的子元素的宽度大小
前端·javascript·css·html
洛千陨1 小时前
element-plus弹窗内分页表格保留勾选项
前端·javascript·vue.js
小小19921 小时前
elementui 单元格添加样式的两种方法
前端·javascript·elementui
完球了1 小时前
【Day02-JS+Vue+Ajax】
javascript·vue.js·笔记·学习·ajax
前端没钱1 小时前
若依Nodejs后台、实现90%以上接口,附体验地址、源码、拓展特色功能
前端·javascript·vue.js·node.js
dgiij1 小时前
AutoX.js向后端传输二进制数据
android·javascript·websocket·node.js·自动化