应用 Optional 的几种模式
创建 Optional 对象
- 声明一个空的Optional
Optional<Car> optCar = Optional.empty();``Optional<Car> optCar = Optional.empty();
- 依据一个非空值创建Optional
你还可以使用静态工厂方法Optional.of,依据一个非空值创建一个Optional对象(如果car是一个null,这段代码会立即抛出一个NullPointerException):
Optional<Car> optCar = Optional.of(car);
- 可接受null的Optional
使用静态工厂方法Optional.ofNullable,你可以创建一个允许null值的Optional对象(如果car是null,那么得到的Optional对象就是个空对象。):
Optional<Car> optCar = Optional.ofNullable(car);
使用 map 从 Optional 对象中提取和转换值
String name = null;
if(insurance != null){
name = insurance.getName();
}
optional写法
Optional<Insurance> optInsurance = Optional.ofNullable(insurance);
Optional<String> name = optInsurance.map(Insurance::getName);
使用 flatMap 链接 Optional 对象
upperName = name.flatMap((value) -> Optional.of(value.toUpperCase()));
System.out.println(upperName.orElse("No value found"));//输出SANAULLA
默认行为及解引用 Optional 对象
- get()是这些方法中最简单但又最不安全的方法。如果变量存在,它直接返回封装的变量值,否则就抛出一个NoSuchElementException异常。所以,除非你非常确定Optional变量一定包含值,否则使用这个方法是个相当糟糕的主意。此外,这种方式即便相对于嵌套式的null检查,也并未体现出多大的改进。
- orElse(T other),它允许你在Optional对象不包含值时提供一个默认值。
- orElseGet(Supplier<? extends T>other)是orElse方法的延迟调用版,Supplier方法只有在Optional对象不含值时才执行调用。如果创建默认值是件耗时费力的工作,你应该考虑采用这种方式(借此提升程序的性能),或者你需要非常确定某个方法仅在Optional为空时才进行调用,也可以考虑该方式(这种情况有严格的限制条件)。
- orElseThrow(Supplier<? extends X> exceptionSupplier)和get方法非常类似,它们遭遇Optional对象为空时都会抛出一个异常,但是使用orElseThrow你可以定制希望抛出的异常类型。
- ifPresent(Consumer<? super T>)让你能在变量值存在时执行一个作为参数传入的方法,否则就不进行任何操作。
两个 Optional 对象的组合
假设你有这样一个方法,它接受一个Person和一个Car对象,并以此为条件对外部提供的服务进行查询,通过一些复杂的业 务逻辑,试图找到满足该组合的最便宜的保险公司:
public Insurance findCheapestInsurance(Person person, Car car) {
// 不同的保险公司提供的查询服务
// 对比所有数据
return cheapestCompany;
}
public Optional<Insurance> nullSafeFindCheapestInsurance(
Optional<Person> person, Optional<Car> car) {
return person.flatMap(p -> car.map(c -> findCheapestInsurance(p, c)));
}
使用 filter 剔除特定的值
Insurance insurance = ...;
if(insurance != null && "CambridgeInsurance".equals(insurance.getName())){
System.out.println("ok");
}
optional写法
Optional<Insurance> optInsurance = ...;
optInsurance.filter(insurance ->"CambridgeInsurance".equals(insurance.getName()))
.ifPresent(x -> System.out.println("ok"));
实例:
用 Optional 封装可能为 null 的值
//Object value = map.get("key");
Optional<Object> value = Optional.ofNullable(map.get("key"));
异常与 Optional 的对比
基础类型的Optional不支持map、flatMap以及filter方法,与Stream一样, Optional对象无法由基础类型的Optional组合构成,所以 Optional 也 提 供 了类 似的 基 础类型——OptionalInt、 OptionalLong以及OptionalDouble等
把所有内容整合起来
@Test
public void test2() {
Properties props = new Properties();
props.setProperty("a", "5");
props.setProperty("b", "true");
props.setProperty("c", "-3");
int i = readDuration(props, "aa");
System.out.println(i);
}
public int readDuration(Properties props, String name) {
return Optional.ofNullable(props.getProperty(name))
.flatMap(OptionalUtility::stringToInt)
.filter(i -> i > 0)
.orElse(0);
}
static class OptionalUtility {
public static Optional<Integer> stringToInt(String s) {
try {
return Optional.of(Integer.parseInt(s));
} catch (NumberFormatException e) {
return Optional.empty();
}
}
}