如何处理null

怎样做才能避免不期而至的NullPointerException呢?通常,可以在需要的地方添加null的检查(过于激进的防御式检查甚至会在不太需要的地方添加检测代码),并且添加的方式往往各有不同。

null-安全的第一种尝试:深层质疑

java8 用Optional取代null-LMLPHP

“深层质疑”,原因是它不断重复着一种模式:每次不确定一个变量是否为null时,都需要添加一个进一步嵌套的if块,也增加了代码缩进的层数。很明显,这种方式不具备扩展性,同时还牺牲了代码的可读性。

null-安全的第二种尝试:过多的退出语句

java8 用Optional取代null-LMLPHP

第二种尝试中,试图避免深层递归的if语句块,采用了一种不同的策略:每次遭遇null变量,都返回一个字符串常量“Unknown”。然而,这种方案远非理想,现在这个方法有了四个截然不同的退出点,使得代码的维护异常艰难。

使用Optional获取car的Insurance名称

汲取Haskell和Scala的灵感,Java 8中引入了一个新的类java.util.Optional<T>。这是一个封装Optional值的类。

public String getCarInsuranceName(Optional<Person> person){
return person.flatMap(Person::getCar)
.flatMap(Car::getInsurance)
.map(Insurance::getName)
.orElse("Unknown");//如果Optional的结果值为空,设置默认值
}

创建Optional对象

1. 声明一个空的Optional

可以通过静态工厂方法Optional.empty,创建一个空的Optional对象:

Optional<Car> optCar = Optional.empty();

2. 依据一个非空值创建Optional

还可以使用静态工厂方法Optional.of,依据一个非空值创建一个Optional对象:

如果car是一个null,这段代码会立即抛出一个NullPointerException,而不是等到试图访问car的属性值时才返回一个错误。

Optional<Car> optCar = Optional.of(car);

3. 可接受null的Optiona

使用静态工厂方法Optional.ofNullable,可以创建一个允许null值的Optional对象:

如果car是null,那么得到的Optional对象就是个空对象。

Optional<Car> optCar = Optional.ofNullable(car);

Optional类的方法

java8 用Optional取代null-LMLPHP

•null引用在历史上被引入到程序设计语言中,目的是为了表示变量值的缺失。

•Java 8中引入了一个新的类java.util.Optional<T>,对存在或缺失的变量值进行建模。

•可以使用静态工厂方法Optional.empty、Optional.of以及Optional.ofNullable创建Optional对象。

•Optional类支持多种方法,比如map、flatMap、filter,它们在概念上与Stream类中对应的方法十分相似。

•使用Optional会迫使更积极地解引用Optional对象,以应对变量值缺失的问题,最终,能更有效地防止代码中出现不期而至的空指针异常。

•使用Optional能帮助设计更好的API,用户只需要阅读方法签名,就能了解该方法是否接受一个Optional类型的值。

附:类代码

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

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

public class Person {
private Optional<Car> car;//人可能有车,也可能没有车,因此将这个字段声明为Optional public Optional<Car> getCar() {
return car;
}
} public class Car {
private Optional<Insurance> insurance;//车可能进行了保险,也可能没有保险,所以将这个字段声明为Optional public Optional<Insurance> getInsurance() {
return insurance;
}
} public class Insurance {
private String name;//保险公司必须有名字 public String getName() {
return name;
}
}

文章来源:java8实战 第十章

05-17 16:11