欢迎访问我的GitHub
这里分类和汇总了欣宸的全部原创(含配套源码):
https://github.com/zq2599/blog_demos
本篇概览
-
本文是《jackson学习》系列第五篇,来熟悉一个常用的注解
JsonInclude
,该注解的仅在序列化操作时有用,用于控制方法、属性等是否应该被序列化;
-
之所以用单独的一篇来写JsonInclude注解,是因为该注解的值有多种,每种都有不同效果,最好的学习方法就是编码实战;
-
先对注解的所有取值做个简介:
ALWAYS // 默认策略,任何情况都执行序列化
NON_NULL // 非空
NON_ABSENT // null的不会序列化,但如果类型是AtomicReference,依然会被序列化
NON_EMPTY // null、集合数组等没有内容、空字符串等,都不会被序列化
NON_DEFAULT // 如果字段是默认值,就不会被序列化
CUSTOM // 此时要指定valueFilter属性,该属性对应一个类,用来自定义判断被JsonInclude修饰的字段是否序列化
USE_DEFAULTS // 当JsonInclude在类和属性上都有时,优先使用属性上的注解,此时如果在序列化的get方法上使用了JsonInclude,并设置为USE_DEFAULTS,就会使用类注解的设置
源码下载
-
如果您不想编码,可以在GitHub下载所有源码,地址和链接信息如下表所示(
https://github.com/zq2599/blog_demos)
:
-
这个git项目中有多个文件夹,本章的应用在
jacksondemo
文件夹下,如下图红框所示:
-
jacksondemo
是父子结构的工程,本篇的代码在
annotation
子工程中,里面的
jsoninclude
这个package下,如下图:
ALWAYS
ALWAYS表示全部序列化,如下图,null和空字符串都会序列化:
NON_NULL
NON_NULL好理解,就是值为null就不序列化:
NON_ABSENT
-
NON_ABSENT略为复杂,当实例化的对象有
Optional
或
AtomicReference
类型的成员变量时,如果Optional引用的实例为空,用
NON_ABSENT
能使该字段不做序列化;
-
Optional是java用来优雅处理空指针的一个特性,本文中不做过多说明,请您自行查阅相关文档;
-
要让Jackson支持Optional特性,必须做两件事,首先是在
pom.xml
中添加以下依赖:
<dependency>
<groupId>com.fasterxml.jackson.datatype</groupId>
<artifactId>jackson-datatype-jdk8</artifactId>
<version>2.11.0</version>
</dependency>
mapper.registerModule(new Jdk8Module());
-
咱们先看看设置成
NON_NULL
时jackson对Optional和AtomicReference的处理,下面的代码中,Optional和AtomicReference的引用都是空,但还是被序列化出来了:
-
代码不变,将
NON_NULL
改为
NON_ABSENT
试试,如下图,可见field2和field3都没有序列化了:
小结NON_ABSENT的效果:
a. 自身为null的字段不会被序列化;
b. Optional类型的字段,如果引用值为null,该字段不会被序列化;
c. AtomicReference类型的字段,如果引用值为null,该字段不会被序列化;
NON_EMPTY
NON_EMPTY好理解,以下情况都不会被序列化:
-
null
-
空字符串
-
空集合
-
空数组
-
Optional类型的,其引用为空
-
AtomicReference类型的,其引用为空
-
演示代码和结果如下图,可见上述场景全部没有被序列化:
NON_DEFAULT
-
设置为NON_DEFAULT后,对保持默认值的字段不做序列化,如下图:
CUSTOM
-
相对其他类型,CUSTOM略为复杂,这个值要配合
valueFilter
属性一起使用;
-
如下所示,JsonInclude的value等于
CUSTOM
时,在序列化的时候会执行CustomFilter的
equals
方法,该方法的入参就是
field0
的值,如果equals方法返回true,
field0就不会被序列化
,如果equals方法返回false时field0才会被序列化
@JsonInclude(value = JsonInclude.Include.CUSTOM,
valueFilter = CustomFilter.class)
private String field0;
-
来看看CustomFilter类的代码,如下所示,只有equals方法,可见:null、非字符串、长度大于2这三种情况都返回true,也就是说这三种情况下都
不会被序列化
:
static class CustomFilter {
@Override
public boolean equals(Object obj) {
// null,或者不是字符串就返回true,意味着不被序列化
if(null==obj || !(obj instanceof String)) {
return true;
// 长度大于2就返回true,意味着不被序列化
return ((String) obj).length() > 2;
}
-
再次强调:valueFilter的equals方法返回true,意味着该字段
不会被序列化!!!
USE_DEFAULTS
-
USE_DEFAULTS的用法也有点绕,咱们通过对比的方法来学习;
-
代码如下所示,在类和成员变量上都有JsonInclude注解,序列化field0的时候,是哪个注解生效呢?:
@JsonInclude(JsonInclude.Include.NON_EMPTY)
static class Test {
@JsonInclude(JsonInclude.Include.NON_NULL)
private List<String> field0;
public List<String> getField0() { return field0; }
public void setField0(List<String> field0) { this.field0 = field0; }
}
-
把field0设置为空集合,运行代码试试,如果类上的注解生效,那么field0就不会被序列化(NON_EMPTY会过滤掉空集合),如果成员变量上的注解生效,field0就会被序列化(NON_NULL只过滤null,空集合不是null),执行结果如下图,可见是成员变量上的注解生效了:
-
接下来保持上述代码不变,仅在
getField0方法
上添加JsonInclude注释,值是
USE_DEFAULTS
,这样在序列化过程中,调用getField0方法时,就用类注解JsonInclude的值了,即NON_EMPTY:
@JsonInclude(JsonInclude.Include.USE_DEFAULTS)
public List<String> getField0() {
return field0;
}
-
执行修改后的代码,如下图所示,此时用的成员变量field0上的注解就不生效了,而是类注解生效,导致空集合不被序列化:
小结
USE_DEFAULTS
的作用如下:
a. 类注解和成员变量注解同时存在时,以成员变量注解为准;
b. 如果对应的get方法也使用了JsonInclude注解,并且值是USE_DEFAULTS,此时以类注解为准;
-
至此,JsonInclude注解的学习和实战就完成了,希望本文能给您提供参考,助您熟练使用注解来指定更精确的序列化过滤策略;
欢迎关注阿里云开发者社区博客:程序员欣宸
学习路上,你不孤单,欣宸原创一路相伴...
BeanWrapper是Spring中一个很重要的接口,Spring在通过配置信息创建对象时,第一步首先就是创建一个BeanWrapper。这篇文章我们就分析下这个接口,本文内容主要对应官网中的3.3及3.4小结
前短时间看了一下狂神的Spring部分的知识,感觉还是有点混乱,并且对于框架部分的学习,还是要靠自己去学习,这样既可以提高自己的自学的能力,还可以帮助自己对Spring有一个更深的记忆,下面主要是通过官方文档,和网上相关资料进行学习总结。