java8-用Optional取代null

Optional的主要作用:预防 Java 中著名的 NullPointerException 异常

NullPointerException 异常

示例:

public class Person {
    private Car car;
}
public class Car {
    private Insurance insurance;
}
public class Insurance {
    private String name;
} 
//会出现
person.getCar().getInsurance().getName();
//采用防御式检查减少NullPointerException
public String getCarInsuranceName(Person person) {
    if (person != null) {
        Car car = person.getCar();
        if (car != null) {
            Insurance insurance = car.getInsurance();
            if (insurance != null) {
                return insurance.getName();
            }
        }
    }
    return "";
}
//减少多次嵌套,但引入过多的退出语句
public String getCarInsuranceName(Person person) {
    if (person == null) {
        return "";
    }
    Car car = person.getCar();
    if (car == null) {
        return "";
    }
    Insurance insurance = car.getInsurance();
    if (insurance == null) {
        return "";
    }
    return insurance.getName();
}

在Java程序开发中使用null会带来理论和实际操作上的种种问题:
它是错误之源。
NullPointerException是目前Java程序开发中最典型的异常。
它会使你的代码膨胀。
它让你的代码充斥着深度嵌套的null检查,代码的可读性糟糕透顶。
它自身是毫无意义的。
null自身没有任何的语义,尤其是,它代表的是在静态类型语言中以一种错误的方式对缺失变量值的建模。
它破坏了Java的哲学。
Java一直试图避免让程序员意识到指针的存在,唯一的例外是:null指针。
它在Java的类型系统上开了个口子。
null并不属于任何类型,这意味着它可以被赋值给任意引用类型的变量。这会导致问题,原因是当这个变量被传递到系统中的另一个部分后,你将无法获知这个null变量最初的赋值到底是什么类型。

解决方法:引入optinal

Groovy语言

Groovy语言通过“安全导航操作符”安全访问可能为null的变量。
def carInsuranceName = person?.car?.insurance?.name

Groovy通过Elvis运算符减少null
//三目运算符“?:”
String displayName = name != null ? name : "Unknown”;
在groovy中
String displayName = name ? name : "Unknown";
String displayName = name ?: "Unknown"

                        `

Scala语言

// Option[T] 是一个类型为 T 的可选值的容器: 如果值存在, Option[T] 就是一个 Some[T] ,如果不存在, Option[T] 就是对象 None
val myMap: Map[String, String] = Map("key1" -> "value")
val value1: Option[String] = myMap.get("key1")
val value2: Option[String] = myMap.get("key2")
 
println(value1) // Some("value1")
println(value2) // None

通过value1.getOrElse("默认值")获取具体值

optional基本用法

创建optional对象

1.声明一个空的Optional
Optional<Car> optCar = Optional.empty();
2.依据一个非空值创建Optional
Optional<Car> optCar = Optional.of(car);
3.可接受null的Optional
Optional<Car> optCar = Optional.ofNullable(car);
  • isPresent ()如果存在返回true,否测返回false。
  • get()是这些方法中最简单但又最不安全的方法。如果变量存在,它直接返回封装的变量值,否则就抛出一个NoSuchElementException异常。所以,除非你非常确定Optional变量一定包含值,否则使用这个方法是个相当糟糕的主意。此外,这种方式即便相对于嵌套式的null检查,也并未体现出多大的改进。
  • orElse(Tother)允许你在Optional对象不包含值时提供一个默认值。
  • orElseGet(Supplier<?extendsT> other)是orElse方法的延迟调用版,Supplier方法只有在Optional对象不含值时才执行调用。如果创建默认值是件耗时费力的工作,你应该考虑采用这种方式(借此提升程序的性能),或者你需要非常确定某个方法仅在Optional为空时才进行调用,也可以考虑该方式(这种情况有严格的限制条件)。
  • orElseThrow(Supplier<?extendsX> exceptionSupplier)和get方法非常类似,它们遭遇Optional对象为空时都会抛出一个异常,但是使用orElseThrow你可以定制希望抛出的异常类型。
  • ifPresent(Consumer<?superT>)让你能在变量值存在时执行一个作为参数传入的方法,否则就不进行任何操作。
       //java7
        String name = null;
        if (insurance != null) {
            name = insurance.getName();
        }

        //不建议这样写,跟java7区别不大
        Optional<Insurance> optInsurance = Optional.ofNullable(insurance);
        String name = null;
        if (optInsurance.isPresent()) {
            name = insurance.get().getName();
        }

        //建议用法
        Optional<Insurance> optInsurance = Optional.ofNullable(insurance);
        String name = optInsurance.map(Insurance::getName).orElse("");
        String name = optInsurance.map(Insurance::getName).orElseThrow(()->new RuntimeException("姓名不能为空"));

        //如果不确定空时的操作,直接返回Optional
        Optional<String> name = optInsurance.map(Insurance::getName);

map/flatmap用法

使用Optional重新定义Person/Car/Insurance的数据模型:

public class Person {
    private Optional<Car> car;
    public Optional<Car> getCar() {  return car;  }
}
public class Car {
    private Optional<Insurance> insurance;
    public Optional<Insurance> getInsurance() {  return insurance;  }
}
public class Insurance {
    private String name;
    public String getName() {  return name;  }
}

获取名称

Optional<Insurance> optInsurance = Optional.ofNullable(insurance); 
Optional<String> name = optInsurance.map(Insurance::getName); 

public String getCarInsuranceName(Optional<Person> person) {
    return person.flatMap(Person::getCar)
                 .flatMap(Car::getInsurance)
                 .map(Insurance::getName)
                 .orElse("Unknown");
}

Stream和Optional的map、flatmap对比

Stream和Optional的map对比

Stream和Optional的map对比.png

如上图:

        Optional<Insurance> optInsurance = Optional.ofNullable(insurance);
        Optional<String> name = optInsurance.map(Insurance::getName);

Stream和Optional中的map操作类似,都是取出里面的具体对象进行操作,再通过Stream或Optional包装后输出。

Stream和Optional的flatmap对比

Optional<Person> person= Optional.ofNullable(Person);
person.map(Person::getCar)//得到的是两层的Optional对象


两层的Optional对象.png

person.flatMap(Person::getCar)//一层的Optional对象


一层的Optional对象.png

Stream和Optional中的flatmap操作类似,都是取出里面的具体对象进行操作,操作的结果是Optional对象或Stream对象,
Optional的话直接输出,Stream的话就将多个Stream合并成一个Stream后输出。
简单的说:可以把Optional看成最多包含一个元素的Stream对象

filter方法

filter用于过滤想要的对象

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,594评论 18 139
  • 关于java8 Optional 文档版本:v1.0版本 和C/C++不一样,java从一开始就尝试将指针彻底的包...
    比轩阅读 3,736评论 1 22
  • Optional 本章内容 如何为缺失的值建模 Optional 类 应用Optional的几种模式 使用Opti...
    追憶逝水年華阅读 1,783评论 0 0
  • 转自: Java 8 中的 Streams API 详解 为什么需要 Stream Stream 作为 Java ...
    普度众生的面瘫青年阅读 2,913评论 0 11
  • 2018.10.4 星期四 晴 国庆节假期快要结束了,这几天皓轩都快玩疯了,背完古诗就是看电...
    皓轩爸爸阅读 114评论 0 2