在写代码的过程中,出现最多的异常可能就是空指针异常了。说白了,空指针异常就是你拿一个不存在的对象,去访问它的成员属性或者方法。我们暂且看下面的代码:
public static String getString() { return null; } public static void main(String[] args) { // 此处返回null String str = getString(); System.err.println(str.length()); } 复制代码
很明显,上面程序运行的结果肯定是排除烦人的 java.lang.NullPointerException
:
Exception in thread "main" java.lang.NullPointerException at org.springframework.web.SpringServletContainerInitializer.main(SpringServletContainerInitializer.java:187) 复制代码
但是,其实空指针本身并不是那么可怕,可怕的是命名我们知道有这么一种可能会出现空指针异常,但是我们却不知道或者没法提前预判,就像上面的代码片段,在eclipse IDE中是没有任何提示的,如下图:
那么有什么办法我们可以提前预判到可能会抛出 NullPointerException
的代码块,然后做防空处理呢!答案是肯定的。下面我将介绍几种常见的防空方式,注意,这不是演习,这不是演习,这不是演习。哈哈!
Optional
下面我将一一举例进行探讨(IDE以eclipse为例):
首先我们要在eclipse启用基于注解的检查,在 Windows->Preference->Java->Compiler->Errors/Warnings->Null Analysis
,如下图所示:
JSR 303/305和Spring都提供了相应的的注解,如下所示:
@Nonnull
、 @Nullable
@NonNull
、 @Nullable
在eclipse启动空检验之后,要想使上面的 @Nullable
注解生效,还需要将注解的全路径类型配置到上图的 Use default annotations for null sercifications
,配置如下图:
配置完成之后,再看下面代码:
/** * 表示返回的值可能为空 * @return */ @Nullable public static String getString() { return null; } /** * 参数不能为空 * @param str * @return */ public static String getName(@NonNull String str) { return null; } public static void main(String[] args) { // 此处返回null String str = getString(); System.err.println(str.length()); // 参数为空 getName(null); } 复制代码
这样子可能为空的代码就一目了然了。当然了,其中的原理还是IDE支持扫描JSR规范的相关注解,Spring的 @NonNull
、 @Nullable
也是基于JSR注解实现的,可用于声明方法,字段或者参数。
Java8提供了对null进行建模的类,它叫 Optional
。我们可以把它理解为一个容器,容器里保存的,可以是方法的放回值,也可以是方法参数值,如果方法返回值为null或者参数值为null,那么就返回一个空容器对象。请看下面代码:
/** * 表示返回的值可能为空 * @return */ public static Optional<String> getString() { // 返回空容器,里边什么也没有 return Optional.empty(); } /** * 参数不能为空 * @param str * @return */ public static String getName(Optional<String> str) { return str.get(); } public static void main(String[] args) { // 此处返回Optional.empty() Optional<String> optional = getString(); // 如果返回的是空容器,那么就返回orElse的值,否则,返回容器里的值 String value = optional.orElse("otherValue"); // 输出otherValue System.err.println(value); // 参数为空 getName(Optional.of("value")); } 复制代码
使用 Optional
类很方便对可能返回null的值进行建模,同时 Optional
里边提供了很多有用的方法允许我们传入一个lambda表达式进行处理,非常方便。