沒(méi)人會(huì)喜歡空指針異常!有什么方法可以避免它們嗎?或許吧。。
本文將討論到以下幾種技術(shù)
1.Optional類(lèi)型(Java 8中新引入的)
2.Objects類(lèi)(Java 7中原有的)
Java 8中的Optional類(lèi)
它是什么?
1.Java 8中新引入的類(lèi)型
2.它是作為某個(gè)指定類(lèi)型的對(duì)象的包裝器或者用于那些不存在對(duì)象(null)的場(chǎng)景
簡(jiǎn)單來(lái)說(shuō),它是處理空值的一個(gè)更好的替代品(警告:乍一看可能并沒(méi)有那么明顯)
基本用法
它是一種類(lèi)型(一個(gè)類(lèi))——那么,怎么才能創(chuàng)建一個(gè)這個(gè)類(lèi)型的實(shí)例?
使用下它的三個(gè)靜態(tài)方法就可以了:
public static Optional<String> stringOptional(String input) {
return Optional.of(input);
}
簡(jiǎn)單明了——創(chuàng)建一個(gè)包含這個(gè)值的Optional包裝器。記住——如果這個(gè)值是null的話,它會(huì)拋出NPE!
public static Optional<String> stringNullableOptional(String input) {
if (!new Random().nextBoolean()) {
input = null;
}
return Optional.ofNullable(input);
}
我個(gè)人認(rèn)為是要更好一點(diǎn)。這樣就不會(huì)有NPE的風(fēng)險(xiǎn)了——如果輸入為null的話,會(huì)返回一個(gè)空的Optional。
public static Optional<String> emptyOptional() {
return Optional.empty();
}
如果你真的就是希望返回一個(gè)”空"值的話。“空”值并不意味著null。
好吧,那如何去消費(fèi)/使用Optional呢?
public static void consumingOptional() {
Optional<String> wrapped = Optional.of("aString");
if (wrapped.isPresent()) {
System.out.println("Got string - " + wrapped.get());
}
else {
System.out.println("Gotcha !");
}
}
簡(jiǎn)單的方法就是檢查Optional包裝器是否真的有值(使用isPresent方法)——你會(huì)懷疑這和使用if(myObj != null)相比有什么好處。別擔(dān)心,這個(gè)我會(huì)解釋清楚的。
public static void consumingNullableOptional() {
String input = null;
if (new Random().nextBoolean()) {
input = "iCanBeNull";
}
Optional<String> wrapped = Optional.ofNullable(input);
System.out.println(wrapped.orElse("default"));
}
你可以使用orElse方法,這樣萬(wàn)一封裝的確實(shí)是一個(gè)null值的話可以用它來(lái)返回一個(gè)默認(rèn)值——它的好處顯而易見(jiàn)。在提取出真實(shí)值的時(shí)候可以避免調(diào)用ifPresent方法這樣明顯多余的方式了。
public static void consumingEmptyOptional() {
String input = null;
if (new Random().nextBoolean()) {
input = "iCanBeNull";
}
Optional<String> wrapped = Optional.ofNullable(input);
System.out.println(wrapped.orElseGet(
() -> {
return "defaultBySupplier";
}
));
}
這個(gè)我就有點(diǎn)搞不清楚了。為什么有兩個(gè)同樣目的的不同方法?orElse和orElseGet明明可以重載的(同名但不同參數(shù))。
不論如何,這兩個(gè)方法明顯的區(qū)別就在于它們的參數(shù)——你可以選擇使用lambda表達(dá)式而不是Supplier的實(shí)例來(lái)完成這個(gè)(一個(gè)函數(shù)式接口)
為什么使用Optional要比常見(jiàn)的null檢查強(qiáng)?
1.使用Optional最大的好處就是可以更明白地表述你的意圖——返回null值的話會(huì)讓消費(fèi)者感到疑惑(當(dāng)真的出現(xiàn)NPE的時(shí)候)這是不是故意返回的,因此還得查看javadoc來(lái)進(jìn)一步定位。而使用Optional就相當(dāng)明了了。
2.有了Optional你就可以徹底避免NPE了——如上所提,使用Optional.ofNullable,orElse以及orElseGet可以讓我們遠(yuǎn)離NPE。
另一個(gè)救星!
看下這個(gè)代碼片段
package com.abhirockzz.wordpress.npesaviors;
import java.util.Map;
import java.util.Objects;
public class UsingObjects {
String getVal(Map<String, String> aMap, String key) {
return aMap.containsKey(key) ? aMap.get(key) : null;
}
public static void main(String[] args) {
UsingObjects obj = new UsingObjects();
obj.getVal(null, "dummy");
}
}
哪個(gè)可能會(huì)為空?
1.Map對(duì)象
2.進(jìn)行搜索使用的key
3.方法調(diào)用的這個(gè)實(shí)例
如果拋出NPE的話,我們?cè)趺茨艽_定到底是哪個(gè)是null的?
package com.abhirockzz.wordpress.npesaviors;
import java.util.Map;
import java.util.Objects;
public class UsingObjects {
String getValSafe(Map<String, String> aMap, String key) {
Map<String, String> safeMap = Objects.requireNonNull(aMap,
"Map is null");
String safeKey = Objects.requireNonNull(key, "Key is null");
return safeMap.containsKey(safeKey) ? safeMap.get(safeKey) : null;
}
public static void main(String[] args) {
UsingObjects obj = new UsingObjects();
obj.getValSafe(null, "dummy");
}
}
requireNonNull方法
1.如果對(duì)象不為null的話就返回它本身
2.如果值為null的話,返回的NPE會(huì)帶有指定的消息
為什么比if(myObj!=null)要好?
你所看到的棧跟蹤信息會(huì)很清楚地看見(jiàn)Objects.requireNonNull的方法調(diào)用。這個(gè)再配合你自己的錯(cuò)誤日志,可以讓你更快地定位問(wèn)題。。。至少在我看來(lái)是更快。
你還可以自己自義校驗(yàn)器,比如說(shuō)實(shí)現(xiàn)一個(gè)簡(jiǎn)單的校驗(yàn)器來(lái)確保沒(méi)有空值。
import java.util.Collections;
import java.util.List;
import java.util.Objects;
import java.util.function.Predicate;
public class RandomGist {
public static <T> T requireNonEmpty(T object, Predicate<T> predicate, String msgToCaller){
Objects.requireNonNull(object);
Objects.requireNonNull(predicate);
if (predicate.test(object)){
throw new IllegalArgumentException(msgToCaller);
}
return object;
}
public static void main(String[] args) {
//Usage 1: an empty string (intentional)
String s = "";
System.out.println(requireNonEmpty(Objects.requireNonNull(s), (s1) -> s1.isEmpty() , "My String is Empty!"));
//Usage 2: an empty List (intentional)
List list = Collections.emptyList();
System.out.println(requireNonEmpty(Objects.requireNonNull(list), (l) -> l.isEmpty(), "List is Empty!").size());
//Usage 3: an empty User (intentional)
User user = new User("");
System.out.println(requireNonEmpty(Objects.requireNonNull(user), (u) -> u.getName().isEmpty(), "User is Empty!"));
}
private static class User {
private String name;
public User(String name){
this.name = name;
}
public String getName(){
return name;
}
}
}
不要讓NPE在錯(cuò)誤的地方成為痛苦。我們有許多工具能更好地處理NPE,甚至徹底地根除它們!