我赌你不清楚Spring中关于Null的这些事

开发 后端
之前一直在某些代码中看到过使用@Nullable 标注过的注释,当时也没有在意到底是什么意思,之后忍不住去调查一番,这篇文章来谈谈Spring中关于Null的那些事。

之前一直在某些代码中看到过使用@Nullable 标注过的注释,当时也没有在意到底是什么意思,之后忍不住去调查一番,这篇文章来谈谈Spring中关于Null的那些事。

在Java中不允许你使用类型表示其null的安全性,但Spring Framework 现在在org.sprinngframework.lang包提供以下注释,以便声明API和字段的可空性:

  •  @Nullable: 用于指定参数、返回值或者字段可以作为null的注释。
  •  @NonNull: 与上述注释相反,表明指定参数、返回值或者字段不允许为null。(不需要@NonNullApi和@NonNullFields适用的参数/返回值和字段)
  •  @NonNullApi: 包级别的注释声明非null作为参数和返回值。
  •  @NonNullFields:包级别的注释声明字段默认非空

Spring Framework 本身利用了上面这几个注释,但它们也可以运用在任何基于Spring的Java 项目中,以声明空安全api 和 空安全字段。尚未支持泛型和数组元素的可空性,但应该也会在后期版本中支持这俩。

Spring Null-Safety出现在Spring5中,让我们更方便的编写空安全的代码,这叫做null-safety,null-safety不是让我们逃脱不安全的代码,而是在编译时产生警告。 此类警告可以在运行时防止灾难性空指针异常(NPE)。

@NonNull

@NonNull注释是null-safety的所有注释中最重要的一个,我们可以使用此注释在期望对象引用的任何地方声明非空约束:字段、方法参数或者方法返回值。

先来看一个例子 

  1. public class Student {  
  2.     private String name;  
  3.     public String getName() {  
  4.         return name;  
  5.     }  
  6.     public void setName(String name) {  
  7.         if(null != null && name.isEmpty()){  
  8.             name = null 
  9.         }  
  10.         this.name = name;  
  11.     }  

上述代码对name的校验是有效的,但是存在一个缺陷,如果name被设置为null的话,那么当我们使用name的时候,就会以NullPointerException来结尾。

使用@NonNull

Spring 的null-safety特性能够允许idea或者eclipse报告这个潜在的威胁,例如,如果我们用IDEA对属性加上@NonNull会出现如下的效果。

奇怪,并没有什么变化啊,没看见有潜在的安全提示啊,那是因为你没有在idea进行设置。

设置安全检查

如果你也没有提示的话,可以通过如下的方式设置安全检查:

如果还不好使的话,那就在右侧 configuration annotations 添加一下 @NonNull和 @Nullable 所在的jar包,如下:

添加上,打上 ✅ 即可看到如下效果。

现在fullName 已经被@NonNull 注释添加编译器检查null值的功能了!

测试一下,可以把@NonNull 注释去掉,你的鼠标再放在fullName 上,就没有这句提示了。

@NonNullFields

@NonNull 注解能够帮助你确保null-safety。然而,如果此注释直接装饰所有的字段的话,就会污染整个代码库。

Spring提供了另外一个不允许为null的注解 — @NonNullFields。这个注解适合用在包级别上,通知我们的开发工具注释包中所有的字段,默认的,不允许为null

新建一个Parent类,并在该类所属包下创建一个名为package-info.java的类,创建的不是Java类,而是创建的file,名为package-info.java,如下

package-info.java 

  1. @NonNullFields  
  2. package com.nullsafety.demo.pojo;  
  3. import org.springframework.lang.NonNullFields; 

新建一个Parent.java 类 

  1. public class Parent {  
  2.     private String son;  
  3.     private String age;  
  4.     private String name;  
  5.     public void setSon(String son) {  
  6.         if(son != null && son.isEmpty()){  
  7.             son = null 
  8.         }  
  9.         this.son = son;  
  10.     }  
  11.     public void setAge(String age) {  
  12.         if(age != null && age.isEmpty()){  
  13.             age = null 
  14.         }  
  15.         this.age = age;  
  16.     }  
  17.     public void setName(String name) {  
  18.         if(name != null && name.isEmpty()){  
  19.             name = null 
  20.         }  
  21.         this.name = name;  
  22.     }  

package-info.java 中的@NonNullFields能够对Parent类中所有的属性起作用,把鼠标放在任意一个属性上,会出现编译期检查的提示

@Nullable

@NonNullFields注释通常比@NonNull更好,因为它有助于减少样板。 但是,有时我们想要从包级别指定的非null约束中免除某些字段,这时候就会使用到@Nullable注解

改造一下Person.java,Person.java 与pack-info.java 处于同一包下 

  1. public class Person {  
  2.     @NonNull  
  3.     private String fullName;  
  4.     @Nullable  
  5.     private String nickName;  
  6.     public String getNickName() {  
  7.         return nickName;  
  8.     }  
  9.     public void setNickName(String nickName) {  
  10.         if(nickName != null && nickName.isEmpty()){  
  11.             nickName = null 
  12.         }  
  13.         this.nickName = nickName;  
  14.     }  
  15.     public String getFullName() {  
  16.         return fullName;  
  17.     }  
  18.     public void setFullName(String fullName) {  
  19.         if(fullName != null && fullName.isEmpty()){  
  20.             fullName = null 
  21.         }  
  22.         this.fullName = fullName;  
  23.     }  

在这种情况下,我们使用@Nullable注释来覆盖字段上@NonNullFields的语义。

@NonNullApi

@NonNullFields注释仅适用于其名称所示的字段。 如果我们想对方法的参数和返回值产生相同的影响,我们需要@NonNullApi。

添加 @NonNullApi和 @NonNullFields 在 configure annotations 中,并选用NonNullApi

与@NonNullFields一样,我们需要在package-info.java 中定义@NonNullApi

package-info.java 

  1. @NonNullApi  
  2. @NonNullFields  
  3. package com.nullsafety.demo.pojo;  
  4. import org.springframework.lang.NonNullApi;  
  5. import org.springframework.lang.NonNullFields; 

加上如下注释后的效果如下: 可以在返回值的时候接受到编译期的提示。

结语

看完文章,你至少应该了解@NonNull, @Nullable, @NonNullFields, @NonNullApi四个注解和各自的作用范围以及如何设置编译期的Null-safety检查。

责任编辑:庞桂玉 来源: Java知音
相关推荐

2021-02-14 22:33:23

Java字符字段

2021-06-07 09:20:56

Javascript运算符开发

2020-09-10 16:10:17

js继承模式前端

2018-05-22 16:24:20

HashMapJavaJDK

2021-03-10 08:56:37

Zookeeper

2020-11-25 09:49:05

Hibernate

2023-08-07 11:07:30

5G电信TECS

2021-07-27 07:31:16

JavaArrayList数组

2022-05-15 21:52:04

typeTypeScriptinterface

2024-02-29 09:08:56

Encoding算法加密

2020-11-11 07:32:18

MySQL InnoDB 存储

2020-10-30 08:20:04

SD卡TF卡存储

2023-07-13 09:05:57

react hook类型types

2018-12-17 12:30:05

Kubernetes存储存储卷

2017-03-07 11:45:57

DevOps容器

2013-12-12 16:59:39

诺基亚微软

2020-03-03 17:35:09

Full GCMinor

2023-02-27 15:46:19

数据元元数据

2021-11-09 06:01:35

前端JITAOT

2023-09-03 21:18:07

Python编程语言
点赞
收藏

51CTO技术栈公众号