一文彻底搞懂 Java 的 Optional|要闻

针对 Java 8 里面的好东西,Optional 就是其中之一,该类提供了一种用于表示可选值而非空引用的类级别解决方案。

01、创建 Optional 对象

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


(资料图片)

Optional<String> 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。

String name = null;Optional<String> optnull = Optional.of(name);

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

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

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

02、判断值是否存在

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

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

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

03、非空表达式

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("为空"));

04、设置(获取)默认值

有时候,我们在创建(获取) 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 引入的语法,方法名后面是没有 ()的,表明该方法并不一定会被调用。

输出结果如下所示:

orElsegetDefaultValueorElseGetgetDefaultValue

输出结果是相似的,没什么太大的不同,这是在 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);}

输出结果如下所示:

orElsegetDefaultValueorElseGet

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

07、获取值

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

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

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

Exception in thread "main" java.util.NoSuchElementException: No value presentat java.base/java.util.Optional.get(Optional.java:141)

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

06、过滤值

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 来完成这个任务,代码该有多冗长。

07、转换值

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

标签:

最近更新

一文彻底搞懂 Java 的 Optional|要闻
2023-03-25 10:53:53
中医药文化进校园_当前播报
2023-03-25 09:01:34
世界今头条!「贵州日报评论员文章」“三问”叩初心 “三干”交答卷
2023-03-25 06:42:55
梦幻西游超级神牛打书
2023-03-25 05:34:49
平南县气象台发布雷电黄色预警信号【III级/较重】
2023-03-25 02:14:17
晚过元老_对于晚过元老简单介绍
2023-03-24 23:00:27
快看:随遇而安反义词有哪些_得过且过随遇而安的反义词
2023-03-24 21:01:47
事关安全!电池热失控一目了然 新资讯
2023-03-24 19:28:29
防溺水教育主题班会教案怎么写_防溺水教育主题班会教案
2023-03-24 17:54:04
郑州3号线地铁站点线路站点(郑州地铁3号线线站点)
2023-03-24 16:20:09
英语ing是什么意思 观速讯
2023-03-24 16:37:29
世界快看:标致2006将于明年推出预售价格为812万元的轿车版本
2023-03-24 15:09:03
快报:秦时明月之沧海横流更新时间什么时候播出_秦时明月之沧海横流更新时间
2023-03-24 13:44:16
源头防范!贵州交警联合交通、应急、文旅部门警示约谈客运企业
2023-03-24 12:20:11
天天看热讯:保利发展退出两家与碧桂园合作公司,碧桂园接盘
2023-03-24 11:08:14
全球热推荐:中南建设为子公司提供融资担保 金额不超过2.16亿元
2023-03-24 10:00:28
喜爱的反义词是什么
2023-03-24 08:59:55
甘肃民乐:戏曲进校园 国粹助成长_世界最新
2023-03-24 08:51:52
全球即时:美股三大股指集体收涨 大型科技股走强 中概股普涨
2023-03-24 07:06:09
绿萝根部发白是怎么回事?
2023-03-24 02:49:37
兴业银行: 兴业银行关于召开2022年度业绩说明会的公告 环球微头条
2023-03-23 23:02:03
人民日报评“多人因造黄谣被处理”:蔑视法律终被法律教训-全球快播报
2023-03-23 20:54:32
电视剧幸福一家人剧情介绍
2023-03-23 19:37:15
天天即时:沪电股份3月23日龙虎榜数据
2023-03-23 19:11:57
当前视讯!信阳市召开社区矫正“厉行法治、护航平安”动员部署会暨重点工作推进会
2023-03-23 17:53:19
鋑联控股(00459.HK)3月23日收盘涨6.25%
2023-03-23 16:44:11
当前滚动:根据巴西政府的一份商业代表团初步名单,近250名巴西企业高管将随该国总统卢拉访华,其中超四分之一来自巴西肉类行业
2023-03-23 15:21:21
漾濞“核妹”闯市场:从卖产品到做品牌
2023-03-23 14:05:26
手写笔亮了!摩托罗拉Moto G Stylus(2023)渲染图曝光-快看点
2023-03-23 12:58:34
环球快播:与众不同的近义词是什么
2023-03-23 12:13:46