使用Optional优雅避免空指针异常

本文已收录至Github,推荐阅读 👉 Java随想录
微信公众号:Java随想录

摘要

空指针异常(NullPointerException)可以说是Java程序员最容易遇到的问题了,影响说大不大,说小也不小。为了解决这个问题,Java 8 版本中推出了 Optional 类。Optional 类是一个容器对象,它可以包含或不包含非空值。使用 Optional 类可以有效地避免空指针异常的问题。在本篇博客中,我将向大家介绍 JDK Optional 类及其使用方法。

Optional 介绍

Optional 类是一个容器对象,它可以包含或不包含非空值。如果一个对象可能为空,那么我们就可以使用 Optional 类来代替该对象。Optional 类型的变量可以有两种状态:存在值和不存在值Optional类有两个重要的方法:of和ofNullable。of方法用于创建一个非空的Optional对象,如果传入的参数为null,则会抛出NullPointerException异常。而ofNullable方法用于创建一个可以为空的Optional对象。如果传入的参数为空,则返回一个空的Optional对象。当 Optional 对象存在值时,调用 get() 方法可以返回该值;当 Optional 对象不存在值时,调用 get() 方法会抛出 NoSuchElementException 异常。

下面是一个使用 Optional 类的例子:

Optional<String> optional = Optional.of("Hello World");
System.out.println(optional.get()); //输出 Hello World

在上面的例子中,我们首先使用 of() 方法创建了一个包含字符串 "Hello World" 的 Optional 对象,然后使用 get() 方法获取该对象的值并将其打印出来。

注意,如果我们尝试创建一个 null 值的 Optional 对象,则会抛出 NullPointerException 异常。而且,在使用 Optional 类时,我们应该尽量避免使用 isPresent() 和 get() 方法,因为这些方法可能会引起空指针异常。比较推荐使用Optional.ofNullable来创建Optional 对象。

Optional 使用

创建 Optional 对象

我们可以使用以下几种方式来创建 Optional 对象:

  1. Optional.of(value):创建一个包含非空值的 Optional 对象。
  2. Optional.empty():创建一个不包含任何值的空 Optional 对象。
  3. Optional.ofNullable(value):创建一个可能包含 null 值的 Optional 对象。如果 value 不为 null,则该方法会创建一个包含该值的 Optional 对象;否则,创建一个空 Optional 对象。

下面是一个使用 Optional.ofNullable() 方法的例子:

String str = null;
Optional<String> optional = Optional.ofNullable(str);
System.out.println(optional.isPresent()); //输出 false

在上面的例子中,我们首先声明了一个空字符串 str,并将其赋值为 null。然后,我们使用 ofNullable() 方法创建了一个 Optional 对象。由于 str 是 null,因此创建的 Optional 对象也是空的。最后,我们使用 isPresent() 方法检查 Optional 对象是否包含值。

orElse()与orElseGet()

orElse()方法接收一个参数,即为默认值。如果Optional对象中的值不为空,则返回该值,否则返回传入的默认值。具体用法如下:

Optional<String> optional = Optional.ofNullable(null);
String result = optional.orElse("default");
System.out.println(result); // 输出 "default"

orElseGet()方法与orElse()方法类似,也是用于获取默认值的方法。但是,orElseGet()方法接收的参数是一个Supplier函数式接口,用于在需要返回默认值时生成该值。具体用法如下:

Optional<String> optional = Optional.ofNullable(null);
String result = optional.orElseGet(() -> "default");
System.out.println(result); // 输出 "default"

orElse() 和 orElseGet()的区别

orElse() 和 orElseGet()特别相似,有必要抽离出来讲下他们之间的区别。

orElse() 方法无论 Optional 对象是否为空都会执行,因此它总是会创建一个新的对象。orElseGet() 方法只有在 Optional 对象为空时才会执行,因此它可以用来延迟创建新的对象。

用一个例子感受一下:

    @Test
    void test() {
        System.out.println("--------不为null的情况----------");
        //不为 null
        String str1 = "hello";
        String result11 = Optional.ofNullable(str1).orElse(get(str1 + ":orElse()方法被执行了"));
        String result12 = Optional.ofNullable(str1).orElseGet(() -> get(str1 + ":orElseGet()方法被执行了"));
        System.out.println(result11);
        System.out.println(result12);
        System.out.println("--------为null的情况----------");
        //为 null
        String str2 = null;
        String result21 = Optional.ofNullable(str2).orElse(get(str1 + ":orElse()方法被执行了"));
        String result22 = Optional.ofNullable(str2).orElseGet(() -> get(str2 + ":orElseGet()方法被执行了"));
        System.out.println(result21);
        System.out.println(result22);
    }


    public String get(String name) {
        System.out.println(name);
        return name;
    }

输出结果如下:

--------不为null的情况----------
hello:orElse()方法被执行了
hello
hello
--------为null的情况----------
hello:orElse()方法被执行了
null:orElseGet()方法被执行了
hello:orElse()方法被执行了
null:orElseGet()方法被执行了

因此,一般来说,如果你希望在 Optional 对象为空时才创建新的对象,可以使用 orElseGet() 方法。否则,如果你希望总是创建新的对象,无论 Optional 对象是否为空,可以使用 orElse() 方法,通常来说orElseGet()更佳,个人也是推荐使用orElseGet()。

map()与flatMap()

map() 方法接受一个函数作为参数,该函数将被应用于 Optional 对象中的值。如果 Optional 对象存在值,则将该值传递给函数进行转换,否则返回一个空 Optional 对象

下面是一个使用 map() 方法的例子:

String str = "Hello";
Optional<String> optional = Optional.of(str);
Optional<String> upperCaseOptional = optional.map(String::toUpperCase);
System.out.println(upperCaseOptional.get()); //输出 HELLO

在上面的例子中,我们首先使用 of() 方法创建了一个包含字符串 "Hello" 的 Optional 对象。然后,我们使用 map() 方法将该字符串转换为大写形式,并将结果存储到另一个 Optional 对象 upperCaseOptional 中。最后,我们使用 get() 方法获取 upperCaseOptional 对象中的值并打印出来。

flatMap() 方法与 map() 方法类似,都接受一个函数作为参数。但是,flatMap() 方法返回的是一个 Optional 类型的值。如果函数返回的是一个 Optional 对象,则该方法会将其“展开”,否则返回一个空 Optional 对象。

下面是一个使用 flatMap() 方法的例子:

String str = "hello world";
Optional<String> optional = Optional.of(str);
Optional<Character> result = optional.flatMap(s -> {
    if (s.length() > 0)
        return Optional.of(s.charAt(0));
    else
        return Optional.empty();
});
System.out.println(result.get()); //输出 h

在上面的例子中,我们首先创建了一个包含字符串 "hello world" 的 Optional 对象。然后,我们使用 flatMap() 方法将该字符串转换为第一个字符,并将结果存储到另一个 Optional 对象 result 中。最后,我们使用 get() 方法获取 result 对象中的值并打印出来。

filter()

filter() 方法接受一个谓词(Predicate)作为参数,返回一个 Optional 类型的值。如果 Optional 对象存在值且满足谓词的条件,则返回该 Optional 对象;否则返回一个空 Optional 对象。

下面是一个使用 filter() 方法的例子:

Integer number = 10;
Optional<Integer> optional = Optional.of(number);
Optional<Integer> result = optional.filter(n -> n % 2 == 0);
System.out.println(result.isPresent()); //输出 true

在上面的例子中,我们首先创建了一个包含整数 10 的 Optional 对象。然后,我们使用 filter() 方法过滤出该数字是否为偶数,并将结果存储到另一个 Optional 对象 result 中。最后,我们使用 isPresent() 方法检查 result 对象是否存在值。

总结

Optional类是Java 8引入的一个非常有用的特性,通过使用 Optional 类,我们可以更加简洁、安全和易读地编写 Java 代码,避免空指针异常等问题。因此,在编写 Java 代码时,我们应该尽可能地使用 Optional 类来处理可能造成空指针的变量。

但是,Optional类并不是万能的,它也有一些不足之处。它可能会增加代码的复杂性和执行时间,因为它需要我们在使用值之前检查是否存在值。因此在使用Optional类时需要谨慎权衡利弊。


本篇文章就到这里,感谢阅读,如果本篇博客有任何错误和建议,欢迎给我留言指正。文章持续更新,可以关注公众号第一时间阅读。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 230,247评论 6 543
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 99,520评论 3 429
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 178,362评论 0 383
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 63,805评论 1 317
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 72,541评论 6 412
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 55,896评论 1 328
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 43,887评论 3 447
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 43,062评论 0 290
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 49,608评论 1 336
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 41,356评论 3 358
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 43,555评论 1 374
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 39,077评论 5 364
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 44,769评论 3 349
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 35,175评论 0 28
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 36,489评论 1 295
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 52,289评论 3 400
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 48,516评论 2 379

推荐阅读更多精彩内容