站内搜索

搜索

06-02 13:26
05-31 17:11

Optional详解

47

主题

38

点数

151

积分

地衡测影

积分
151

柴到了

发表于 2025-3-21 02:10:47 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。

您需要 登录 才可以下载或查看,没有账号?立即注册

x

面试官:你了解Optional嘛?

​ 首先: 什么是Optional? 它是Java8中为了解决空指针问题而产生的。该类提供了一种用于表示可选值而非空引用的类级别的解决方案。

​ 我们程序员为了避免NullPointerException,于是在代码中就有了大量的null值检查。尽管有时候这种检查完全没用必要,而且影响代码的美感。但是我们已经习惯了这样解决。Optional的引用就是为了以便我们编写的代码不再那么呆板。

1.没有Optional类会有什么问题?

​ 我们先模拟一个实际的应用场景, 小李需要从数据库中根据会员 ID 拉取一个会员的姓名,然后将姓名打印到控制台。它的解决方法如下:

package com.alex;

/**
 * @program: optionalDemo
 * @description: 无Optional的代码示例,会员不存在时,会抛出空指针异常,需要手动判断是否为空,代码冗余,可读性差,不推荐使用
 * @author: Alex
 * @create: 2023-07-05 15:59
 **/

public class WithoutOptionalDemo {
    class Member{
        private String name;

        public String getName() {
            return name;
        }

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

    public static void main(String[] args) {
        Member mem = findMemberById();
        // 判断会员是否存在
        if (mem != null) {
            System.out.println(mem.getName());
        }
    }

    public static Member findMemberById(){
        // 当前id的会员不存在
        return null;
    }
}

​ 由于当前ID的会员不存在,所以findMemberById方法返回null,如果在打印会员信息之前,不做非空判断,那么就会抛出NPE异常

Exception in thread "main" java.lang.NullPointerException
    at com.alex.WithoutOptionalDemo.main(WithoutOptionalDemo.java:28)

2.Optional是如何解决这个问题的?

​ 小李通过Optional代码进行了优化,我们来看看小李是然后优化的上述问题的。

package com.alex;

import java.util.Optional;

/**
 * @program: optionalDemo
 * @description: Optional类的使用示例
 * @author: Alex
 * @create: 2023-07-05 16:11
 **/

public class OptionalDemo {
    public static void main(String[] args) {
        Optional<Member> optional = findMemberById();
        optional.ifPresent(mem -> {
            System.out.println("会员姓名是:" + mem.getName());
        });
    }
    public static Optional<Member> findMemberById(){
        // 从数据库中查找id对应的Member对象
        // 如果找到了,返回Optional.of(member)
        // 如果没找到,返回Optional.empty()
        boolean hasName = true;
        if (hasName){
            return Optional.of(new Member("Alex"));
        }
        return Optional.empty();
    }
}

 class Member{
    private String name;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
     public Member() {
     }
    public Member(String name) {
        this.name = name;
    }
}

findMemberById方法返回了Optional<Member> 作为结果,这样就表明 Member 可能存在,也可能不存在,这时候就可以在 Optional 的 ifPresent() 方法中使用 Lambda 表达式来直接打印结果。

​ Optional 之所以可以解决 NPE 的问题,是因为它明确的告诉我们,不需要对它进行判空。它就好像十字路口的路标,明确地告诉你该往哪走。

3.创建Optional对象

1)可以使用静态方法 empty() 创建一个空的 Optional 对象

Optional<Object> empty = Optional.empty();
System.out.println(empty); // 输出:Optional.empty

2)可以使用静态方法 of() 创建一个非空的 Optional 对象

Optional<String> opt = Optional.of("小李");
System.out.println(opt); // 输出:Optional[小李]

当然了,传递给 of() 方法的参数必须是非空的,也就是说不能为 null,否则仍然会抛出 NullPointerException。

3)可以使用静态方法 ofNullable() 创建一个即可空又可非空的 Optional 对象

String name = null;
Optional<String> optOrNull = Optional.ofNullable(name);
System.out.println(optOrNull); // 输出:Optional.empty

ofNullable() 方法内部有一个三元表达式,如果为参数为 null,则返回私有常量 EMPTY;否则使用 new 关键字创建了一个新的 Optional 对象——不会再抛出 NPE 异常了。

4.判断值是否存在

​ 可以通过方法 isPresent() 判断一个 Optional 对象是否存在,如果存在,该方法返回 true,否则返回 false——取代了 obj != null 的判断。

Optional<String> opt = Optional.of("小李");
System.out.println(opt.isPresent()); // 输出:true

Optional<String> optOrNull = Optional.ofNullable(null);
System.out.println(opt.isPresent()); // 输出:false

Java 11 后还可以通过方法 isEmpty() 判断与 isPresent() 相反的结果。

Optional<String> opt = Optional.of("小李");
System.out.println(opt.isPresent()); // 输出:false

Optional<String> optOrNull = Optional.ofNullable(null);
System.out.println(opt.isPresent()); // 输出:true

5.非空表达式

​ Optional 类有一个非常现代化的方法——ifPresent(),允许我们使用函数式编程的方式执行一些代码,因此,我把它称为非空表达式。如果没有该方法的话,我们通常需要先通过 isPresent() 方法对 Optional 对象进行判空后再执行相应的代码:

Optional<String> optOrNull = Optional.ofNullable(null);
if (optOrNull.isPresent()) {
    System.out.println(optOrNull.get().length());
}

​ 有了 ifPresent() 之后,情况就完全不同了,可以直接将 Lambda 表达式传递给该方法,代码更加简洁,更加直观。

Optional<String> opt = Optional.of("小李");
opt.ifPresent(str -> System.out.println(str.length()));

​ Java 9 后还可以通过方法 ifPresentOrElse(action, emptyAction) 执行两种结果,非空时执行 action,空时执行 emptyAction。

Optional<String> opt = Optional.of("小李");
opt.ifPresentOrElse(str -> System.out.println(str.length()), () -> System.out.println("为空"));

6.设置(获取)默认值

​ 有时候,我们在创建(获取) Optional 对象的时候,需要一个默认值,orElse()orElseGet() 方法就派上用场了。

orElse() 方法用于返回包裹在 Optional 对象中的值,如果该值不为 null,则返回;否则返回默认值。该方法的参数类型和值得类型一致。

String nullName = null;
String name = Optional.ofNullable(nullName).orElse("小李");
System.out.println(name); // 输出:小李

orElseGet() 方法与 orElse() 方法类似,但参数类型不同。如果 Optional 对象中的值为 null,则执行参数中的函数。

String nullName = null;
String name = Optional.ofNullable(nullName).orElseGet(()->"小李");
System.out.println(name); // 输出:小李

​ 从输出结果以及代码的形式上来看,这两个方法极其相似,这不免引起我们的怀疑,Java 类库的设计者有必要这样做吗?

​ 假设现在有这样一个获取默认值的方法,很传统的方式。

public static String getDefaultValue() {
    System.out.println("getDefaultValue");
    return "小李";
}

​ 然后,通过 orElse() 方法和 orElseGet() 方法分别调用 getDefaultValue() 方法返回默认值。

public static void main(String[] args) {
    String name = null;
    System.out.println("orElse");
    String name2 = Optional.ofNullable(name).orElse(getDefaultValue());

    System.out.println("orElseGet");
    String name3 = Optional.ofNullable(name).orElseGet(OrElseOptionalDemo::getDefaultValue);
}

注:类名 :: 方法名是 Java 8 引入的语法,方法名后面是没有 () 的,表明该方法并不一定会被调用。

输出结果如下所示:

orElse
getDefaultValue

orElseGet
getDefaultValue

​ 输出结果是相似的,没什么太大的不同,这是在 Optional 对象的值为 null 的情况下。假如 Optional 对象的值不为 null 呢?

public static void main(String[] args) {
    String name = "小李";
    System.out.println("orElse");
    String name2 = Optional.ofNullable(name).orElse(getDefaultValue());

    System.out.println("orElseGet");
    String name3 = Optional.ofNullable(name).orElseGet(OrElseOptionalDemo::getDefaultValue);
}

输出结果如下所示:

orElse
getDefaultValue
orElseGet

咦,orElseGet() 没有去调用 getDefaultValue()。哪个方法的性能更佳,你明白了吧?

7.获取值

​ 直观从语义上来看,get() 方法才是最正宗的获取 Optional 对象值的方法,但很遗憾,该方法是有缺陷的,因为假如 Optional 对象的值为 null,该方法会抛出 NoSuchElementException 异常。这完全与我们使用 Optional 类的初衷相悖。

public class GetOptionalDemo {
    public static void main(String[] args) {
        String name = null;
        Optional<String> optOrNull = Optional.ofNullable(name);
        System.out.println(optOrNull.get());
    }
}

​ 这段程序在运行时会抛出异常:

Exception in thread "main" java.util.NoSuchElementException: No value present
    at java.base/java.util.Optional.get(Optional.java:141)
    at com.cmower.dzone.optional.GetOptionalDemo.main(GetOptionalDemo.java:9)

​ 尽管抛出的异常是 NoSuchElementException 而不是 NPE,但在我们看来,显然是在“五十步笑百步”。建议 orElseGet() 方法获取 Optional 对象的值。

8.过滤值

​ 小李通过 Optional 类对之前的代码进行了升级,完成后又兴高采烈地跑去找了个任务了:用户注册时对密码的长度进行检查。

​ 小李找到任务后,乐开了花,因为他刚要学习 Optional 类的 filter() 方法,这就派上了用场。

public class FilterOptionalDemo {
    public static void main(String[] args) {
        String password = "12345";
        Optional<String> opt = Optional.ofNullable(password);
        System.out.println(opt.filter(pwd -> pwd.length() > 6).isPresent());
    }
}

filter() 方法的参数类型为 Predicate(Java 8 新增的一个函数式接口),也就是说可以将一个 Lambda 表达式传递给该方法作为条件,如果表达式的结果为 false,则返回一个 EMPTY 的 Optional 对象,否则返回过滤后的 Optional 对象。

​ 在上例中,由于 password 的长度为 5 ,所以程序输出的结果为 false。假设密码的长度要求在 6 到 10 位之间,那么还可以再追加一个条件。来看小王增加难度后的代码。

Predicate<String> len6 = pwd -> pwd.length() > 6;
Predicate<String> len10 = pwd -> pwd.length() < 10;

password = "1234567";
opt = Optional.ofNullable(password);
boolean result = opt.filter(len6.and(len10)).isPresent();
System.out.println(result);

​ 这次程序输出的结果为 true,因为密码变成了 7 位,在 6 到 10 位之间。想象一下,假如小李使用 if-else 来完成这个任务,代码该有多冗长。

9.转换值

​ 小李检查完了密码的长度,仍然觉得不够尽兴,觉得要对密码的强度也进行检查,比如说密码不能是“password”,这样的密码太弱了。于是他又开始研究起了 map() 方法,该方法可以按照一定的规则将原有 Optional 对象转换为一个新的 Optional 对象,原有的 Optional 对象不会更改。

先来看小李写的一个简单的例子:

public class OptionalMapDemo {
    public static void main(String[] args) {
        String name = "小李";
        Optional<String> nameOptional = Optional.of(name);
        Optional<Integer> intOpt = nameOptional
                .map(String::length);

        System.out.println( intOpt.orElse(0));
    }
}

​ 在上面这个例子中,map() 方法的参数 String::length,意味着要 将原有的字符串类型的 Optional 按照字符串长度重新生成一个新的 Optional 对象,类型为 Integer。

​ 搞清楚了 map() 方法的基本用法后,小李决定把 map() 方法与 filter() 方法结合起来用,前者用于将密码转化为小写,后者用于判断长度以及是否是“password”。

public class OptionalMapFilterDemo {
    public static void main(String[] args) {
        String password = "password";
        Optional<String>  opt = Optional.ofNullable(password);

        Predicate<String> len6 = pwd -> pwd.length() > 6;
        Predicate<String> len10 = pwd -> pwd.length() < 10;
        Predicate<String> eq = pwd -> pwd.equals("password");

        boolean result = opt.map(String::toLowerCase).filter(len6.and(len10 ).and(eq)).isPresent();
        System.out.println(result);
    }
}

​ 以上就是本文的全部内容了---Optional 指南

温馨提示:看帖回帖是一种美德,您的每一次发帖、回帖都是对论坛最大的支持,谢谢! [这是默认签名,点我更换签名]
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

加入社群

加入社群

Pixtech

Powered by Pixtech

© 2025 Pixtech Team.