Optional类

Optional类

Optional<T>类是一个容器类,代表一个值存在和不存在,原来用不惯null表示一个值的不存在,现在用Optional可以更好的表达这个概念;是Java1.8之后解决null值判断问题。

常用方法列表

method desc
Optional.of(T t) 创建一个非空值的Optional实例
Optional.empty() 创建一个空的Optional实例
Optional.ofNullable(T t) 若t不为null,创建Optional实例,否则创建空实例
isPresent() 判断是否包含值
ifPresent(Consumer c) 如果存在值,则使用该值调用指定的消费者,否则不执行任何操作
get() 如果 Optional中有值,返回值,否则抛出 NoSuchElementException
orElse(T t) 如果对象包含值,返回该值,否则返回t
orElseGet(Supplier s) 如果对象包含值,返回该值,否则返回s中获取的值
orElseThrow(Supplier s) 如果容器为空,返回自定义异常信息
filter(Predicate p) 断言型接口返回true,则返回Optional描述的值;否则返回Optional.empty()
map(Function f) 如果有值对其处理,并返回处理后的Optional,否则返回Optional.empty()
flatmap(Function mapper) 与map类似,返回值要求必须是Optional

引入

看一下下面的代码,为了避免空指针异常,我们需要多层的if判断,不利于代码的易读性同时也不利于书写。

    if (people != null) {
        Country country = people.getCountry();
        if (country != null) {
            City city = country.getCity();
            if (city != null) {
                Street street = city.getStreet();
                if (street != null) {
                    String streetName = street.getStreetName();
                }
            }
        }
    }

创建实例方式

  • Optional.of(T t)

        public void test02() {
            Optional<People> p = Optional.of(new People());
            People people = p.get();
        }
    

    Optional.of(null),无法创建实例,导致NullPointerException

  • Optional.empty()

        public void test03() {
            Optional<People> p = Optional.empty();
            People people = p.get();
        }
    

    创建Optional的空实例,若使用p.get()获取对象,会导致java.util.NoSuchElementException: No value present

  • Optional.ofNullable(T t)

        public void test04() {
            Optional<People> p = Optional.ofNullable(new People());
    
            People people = p.get();
            System.out.println(people);
        }
    

    使用Optional.ofNullable构建Optional实例时,参数可以为其他实例,也可以为null,如果为null时,默认返回的是Optional.empty

          // 源码
        public static <T> Optional<T> ofNullable(T value) {
            return value == null ? empty() : of(value);
        }
    

获取Optional的值

    public void test04() {
        Optional<People> p = Optional.ofNullable(new People());

        People people = p.get();
        System.out.println(people);
    }

Optional容器类通过get()方法获取容器中的对象,但若是Optional是个空实例,就会发生NoSuchElementException异常,所以Optional容器类提供了isPresent()方法来判断Optional是否是空实例。

    public void test04() {
        Optional<People> p = Optional.ofNullable(new People());

        if (p.isPresent()) {
            People people = p.get();
            System.out.println(people);
        }
    }
    public void test05() {
        String name = "zhangsan";
        Optional<String> optional = Optional.ofNullable("zhangsan");

        Assert.assertTrue(optional.isPresent());

        Assert.assertEquals(name, optional.get());
    }

检查Optional容器类是否有值另外一个方法ifPresent(Consumer c); 如果有值则调用消费者代码块,否则不执行任何操作。若有值而Consumer为空,则会导致NullPointerException

    public void test06() {
        Optional<String> optional = Optional.ofNullable("zhangsan");
        optional.ifPresent(System.out::println);
    }

兜底函数

兜底函数可以理解为默认值,当Optional容器中没有值,则会返回兜底函数的值,避免获取不到值的情况

orElse(T t)

如果对象包含值,返回该值,否则返回t (注意:返回的是具体的值,而不是Optional容器类)

    public void test07() {
        String name = null;
        String username = Optional.ofNullable(name).orElse("zhangsan");

        Assert.assertEquals("zhangsan", username);
    }

orElseGet(Supplier s)

如果对象包含值,返回该值,否则执行Supplier代码块

public void test08() {
    String name = null;
    People people = new People("zhangsan", 18, null);
    String username = Optional.ofNullable(name).orElseGet(people::getName);

    Assert.assertEquals("zhangsan", username);
}

orElse和orElseGet对比

若传入的值为null时

    @Test
    public void test09() {
        System.out.println("orElse");
        String name1 = (String) Optional.ofNullable(null).orElse(getName());

        System.out.println("orElseGet");
        String name2 = (String) Optional.ofNullable(null).orElseGet(() -> getName());
    }

    public String getName() {
        System.out.println("getName");
        return "zhangsan";
    }

结果输出

orElse
getName
orElseGet
getName

可以看出,两者并没有差异,都会调用getName()方法

若传入非null值

    public void test10() {
        System.out.println("orElse");
        String name1 = (String) Optional.ofNullable("lisi").orElse(getName());

        System.out.println("orElseGet");
        String name2 = (String) Optional.ofNullable("lisi").orElseGet(() -> getName());
    }

结果输出

orElse
getName
orElseGet

若传入非null值时,orElse()还是会调用getName(),而orElseGet()则不会执行。

在密集型调用,或者需求量较大时,两者性能上的差异显而易见。

orElseThrow(Supplier s)

如果Optional容器为空,可以返回自定义的异常信息

    public void test11() {
        Object o = Optional.ofNullable(null).orElseThrow(IllegalArgumentException::new);
    }

值过滤

filter(Predicate p)

通过filter(Predicate p)方法过滤出来符合要求的值,其中当断言型接口p返回true,则返回Optional描述的值;否则返回空的Optional

    public void test12() {
        People p = new People("zhangsan", 11, null);
        Optional<People> people = Optional.ofNullable(p)
                .filter(x -> x.getAge() == 11);

        Assert.assertTrue(people.isPresent());
    }

值转换

map()

如果有值对其处理,并返回处理后的Optional,否则返回Optional.empty();

    public void test13() {
        People p = new People("zhangsan", 11, null);

        Optional<String> optionalS = Optional.ofNullable(p).map(x -> x.getName());

        Assert.assertTrue(optionalS.isPresent());
        
        Assert.assertEquals("zhangsan", optionalS.get());
    }

flatmap(Function mapper)

与map类似,返回值要求必须是Optional

    public void test14() {
        People p = new People("zhangsan", 11, null);

        Optional<String> optionalS = Optional.ofNullable(p).flatMap(x -> Optional.ofNullable(x.getName()));

        Assert.assertTrue(optionalS.isPresent());

        Assert.assertEquals("zhangsan", optionalS.get());
    }

总结

链式编程

filter,map,fliterMap返回都是Optional描述的值,所以就可以使用链式编程

    public void test15() {
        People p = null;

        String str = Optional.ofNullable(p)
                .filter(x -> x.getName() != null)
                .flatMap(x -> Optional.ofNullable(x.getName()))
                .orElse("lisi");

        Assert.assertEquals(str, "lisi");
    }

引入的优化

// 每个属性值都是用Optional封装
private Optional<Country> country;
    public void test20() {
        People p = null;
        String s = Optional.ofNullable(p)
                .flatMap(x -> x.getCountry())
                .flatMap(x -> x.getCity())
                .flatMap(x -> x.getStreet())
                .map(x -> x.getStreetName())
                .orElse("北京东路");

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