今天再看阿里的Java开发手册,里面异常处理第10条提到这样一个建议。程序员
【推荐】防止 NPE ,是程序员的基本修养,注意 NPE 产生的场景:
1 ) 返回类型为基本数据类型,return 包装数据类型的对象时,自动拆箱有可能产生 NPE。
反例: public int f() { return Integer 对象}, 若是为 null ,自动解箱抛 NPE 。
2 ) 数据库的查询结果可能为 null 。
3 ) 集合里的元素即便 isNotEmpty ,取出的数据元素也可能为 null 。
4 ) 远程调用返回对象时,一概要求进行空指针判断,防止 NPE 。
5 ) 对于 Session 中获取的数据,建议 NPE 检查,避免空指针。
6 ) 级联调用 obj . getA() . getB() . getC(); 一连串调用,易产生 NPE 。
正例:使用 JDK8 的 Optional 类来防止 NPE 问题。数据库
里面的正确示例提示咱们用Java8的Optional类来防止NPE的问题。app
那咱们今天就看看这个Optional类吧函数
序号 | 方法 & 描述 |
---|---|
1 | static <T> Optional<T> empty() 返回空的 Optional 实例。ui |
2 | boolean equals(Object obj) 判断其余对象是否等于 Optional。spa |
3 | Optional<T> filter(Predicate<? super <T> predicate) 若是值存在,而且这个值匹配给定的 predicate,返回一个Optional用以描述这个值,不然返回一个空的Optional。指针 |
4 | <U> Optional<U> flatMap(Function<? super T,Optional<U>> mapper) 若是值存在,返回基于Optional包含的映射方法的值,不然返回一个空的Optional调试 |
5 | T get() 若是在这个Optional中包含这个值,返回值,不然抛出异常:NoSuchElementExceptioncode |
6 | int hashCode() 返回存在值的哈希码,若是值不存在 返回 0。对象 |
7 | void ifPresent(Consumer<? super T> consumer) 若是值存在则使用该值调用 consumer , 不然不作任何事情。 |
8 | boolean isPresent() 若是值存在则方法会返回true,不然返回 false。 |
9 | <U>Optional<U> map(Function<? super T,? extends U> mapper) 若是有值,则对其执行调用映射函数获得返回值。若是返回值不为 null,则建立包含映射返回值的Optional做为map方法返回值,不然返回空Optional。 |
10 | static <T> Optional<T> of(T value) 返回一个指定非null值的Optional。 |
11 | static <T> Optional<T> ofNullable(T value) 若是为非空,返回 Optional 描述的指定值,不然返回空的 Optional。 |
12 | T orElse(T other) 若是存在该值,返回值, 不然返回 other。 |
13 | T orElseGet(Supplier<? extends T> other) 若是存在该值,返回值, 不然触发 other,并返回 other 调用的结果。 |
14 | <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier)
若是存在该值,返回包含的值,不然抛出由 Supplier 继承的异常 |
15 | String toString() 返回一个Optional的非空字符串,用来调试 |
咱们能够看到Optional总共也就10+个方法,其中有三个static方法。而且Optional的构造方法是private,不能new出来。
因此咱们通常用这三个static来获取Optional的对象。
1 public static <T> Optional<T> of(T value) { 2 return new Optional<>(value); 3 } 4 5 public static <T> Optional<T> ofNullable(T value) { 6 return value == null ? empty() : of(value); 7 }
很明显 of 对null对象没有作任何处理,ofNullable才作了处理。因此当咱们不知道传入的对象是否为null的时候,咱们应该选择用 ofNullable来作处理。
1 public<U> Optional<U> map(Function<? super T, ? extends U> mapper) { 2 Objects.requireNonNull(mapper); 3 if (!isPresent()) 4 return empty(); 5 else { 6 return Optional.ofNullable(mapper.apply(value)); 7 } 8 }
若是咱们想获取Object里面的值的话,咱们就须要用到这个map。
1 public class OptionalTest { 2 public static void main(String[] args) { 3 Person person = new Person("zhangsan", 18); 4 String name = getName(person); 5 System.out.println(name); 6 } 7 8 private static String getName(Person person) { 9 if (Objects.isNull(person)) { 10 return "unKnown"; 11 } 12 return person.getName(); 13 } 14 15 }
咱们看上面的这个例子。
咱们有一个函数 getName 做用是获取Person对象的名字。但我并不知道这个Person是否为Null。
因此我要进行一个判断,判断Person是否为空,在作决定。
但若是咱们使用Optional类的话,咱们能够这样写
1 public class OptionalTest { 2 public static void main(String[] args) { 3 Person person = new Person("zhangsan", 18); 4 String name = getName(person); 5 System.out.println(name); 6 } 7 8 private static String getName(Person person) { 9 String name = Optional.ofNullable(person).map(x -> x.getName()) 10 .orElse("unKnown"); 11 return name; 12 } 13 }
若是传入的为空,它会自动new一个 Optional<T> t = (Optional<T>) EMPTY;
有效的处理到了null的问题,并且还很是的简洁。