写点什么

jackson 学习之八:常用方法注解

作者:程序员欣宸
  • 2022 年 4 月 13 日
  • 本文字数:3457 字

    阅读完需:约 11 分钟

jackson学习之八:常用方法注解

欢迎访问我的 GitHub

这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos

本篇概览

  • 本文是《jackson 学习》系列的第八篇,继续学习 jackson 强大的注解能力,本篇学习常用的方法注解,并通过实例来加深印象,下图是常用方法注解的简介:

源码下载

  1. 如果您不想编码,可以在 GitHub 下载所有源码,地址和链接信息如下表所示(https://github.com/zq2599/blog_demos):



  1. 这个 git 项目中有多个文件夹,本章的应用在 jacksondemo 文件夹下,如下图红框所示:

  2. jacksondemo 是父子结构的工程,本篇的代码在 annotation 子工程中,里面的 methodannotation 这个 package 下,如下图:


JsonValue

  1. 在序列化时起作用,可以用来注解 get 方法或者成员变量;

  2. 一个类中,JsonValue 只允许出现一次;

  3. 如果注解的是 get 方法,那么该方法的返回值就是整个实例的序列化结果;

  4. 如果注解的是成员变量,那么该成员变量的值就是整个实例的序列化结果;

  5. 下面是用来测试的 Pojo 类,JsonValue 注解放在 getField0 方法上,此方法的返回值已经写死了"abc":

    static class Test {
private String field0;
private String field1;
@JsonValue public String getField0() { return "abc"; }
public void setField0(String field0) { this.field0 = field0; } public String getField1() { return field1; } public void setField1(String field1) { this.field1 = field1; } }
复制代码
  1. Test 类的序列化结果如下,即 getField0 方法的返回值:

JsonCreator

  1. 在反序列化时,当出现有参构造方法时(可能是多个有参构造方法),需要通过 JsonCreator 注解指定反序列化时用哪个构造方法,并且在入参处还要通过 JsonProperty 指定字段关系:

    static class Test {
private String field0; private String field1;

public Test(String field0) { this.field0 = field0; }
// 通过JsonCreator指定反序列化的时候使用这个构造方法 // 通过JsonProperty指定字段关系 @JsonCreator public Test(@JsonProperty("field0") String field0, @JsonProperty("field1") String field1) { this.field0 = field0; this.field1 = field1; }
@Override public String toString() { return "Test{" + "field0='" + field0 + '\'' + ", field1='" + field1 + '\'' + '}'; } }
复制代码
  1. 反序列化结果如下:

JsonSetter

  1. JsonSetter 注解在 set 方法上,被用来在反序列化时指定 set 方法对应 json 的哪个属性;

  2. JsonSetter 源码中,推荐使用 JsonProperty 来取代 JsonSetter:


  1. 测试代码和结果如下,可见反序列化时,是按照 JsonSetter 的 value 去 json 中查找属性的:

JsonGetter

  1. JsonGetter 只能作为方法注解;

  2. 在序列化时,被 JsonGetter 注解的 get 方法,对应的 json 字段名是 JsonGetter 的 value;

  3. JsonGetter 源码中,推荐使用 JsonProperty 来取代 JsonGetter:

  4. 测试代码和结果如下,可见序列化时 JsonGetter 的 value 会被作为 json 字段名:

JsonAnyGetter

  1. JsonAnyGetter 的作用有些特别:在序列化时,用 Map 对象的键值对转成 json 的字段和值;

  2. 理解 JsonAnyGetter 最好的办法,是对比使用前后序列化结果的变化,先来看以下这段代码,是没有 JsonAnyGetter 注解的,Test 有两个成员变量,其中 map 字段是 HashMap 类型的:

package com.bolingcavalry.jacksondemo.annotation.methodannotation;
import com.fasterxml.jackson.annotation.JsonAnyGetter;import com.fasterxml.jackson.databind.ObjectMapper;import com.fasterxml.jackson.databind.SerializationFeature;
import java.util.HashMap;import java.util.Map;
public class JsonAnySetterSerialization {
static class Test { private String field0; private Map<String, Object> map;
public String getField0() { return field0; } public void setField0(String field0) { this.field0 = field0; } public void setMap(Map<String, Object> map) { this.map = map; } public Map<String, Object> getMap() { return map; } }
public static void main(String[] args) throws Exception { ObjectMapper mapper = new ObjectMapper(); // 美化输出 mapper.enable(SerializationFeature.INDENT_OUTPUT);
// 新增一个HashMap,里面放入两个元素 Map<String, Object> map = new HashMap<>(); map.put("aaa", "value_aaa"); map.put("bbb", "value_bbb");
Test test = new Test(); test.setField0("000");
// map赋值给test.map test.setMap(map);
System.out.println(mapper.writeValueAsString(test)); }}
复制代码


  1. 上述代码的执行结果如下,其实很好理解,就是 field0 和 map 两个字段而已:

{  "field0" : "000",  "map" : {    "aaa" : "value_aaa",    "bbb" : "value_bbb"  }}
复制代码
  1. 接下来,对上述代码做一处改动,如下图红框所示,给 getMap 方法增加 JsonAnyGetter 注解:

  2. 修改后的执行结果如下,原来的 map 字段没有了,map 内部的所有键值对都成了 json 的字段:

{  "field0" : "000",  "aaa" : "value_aaa",  "bbb" : "value_bbb"}
复制代码


  1. 至此,可以品味出 JsonAnyGetter 的作用了:序列化时,将 Map 中的键值对全部作为 JSON 的字段输出

JsonAnySetter

  1. 弄懂了前面的 JsonAnyGetter,对于 JsonAnySetter 的作用想必您也能大致猜到:反序列化时,对 json 中不认识的字段,统统调用 JsonAnySetter 注解修饰的方法去处理;

  2. 测试的代码如下,Test 类的 setValue 方法被 JsonAnySetter 注解,在反序列化时,json 中的 aaa bbb 字段,都会交给 setValue 方法处理,也就是放入 map 中:

package com.bolingcavalry.jacksondemo.annotation.methodannotation;
import com.fasterxml.jackson.annotation.JsonAnySetter;import com.fasterxml.jackson.annotation.JsonCreator;import com.fasterxml.jackson.annotation.JsonProperty;import com.fasterxml.jackson.databind.ObjectMapper;
import java.util.HashMap;import java.util.Map;
public class JsonAnySetterDeserialization {
static class Test {
private String field0; private Map<String, Object> map = new HashMap<>();
@JsonAnySetter public void setValue(String key, Object value) { map.put(key, value); }
@Override public String toString() { return "Test{" + "field0='" + field0 + '\'' + ", map=" + map + '}'; } }
public static void main(String[] args) throws Exception { String jsonStr = "{\n" + " \"field0\" : \"000\",\n" + " \"aaa\" : \"value_aaa\",\n" + " \"bbb\" : \"value_bbb\"\n" + "}";
System.out.println(new ObjectMapper().readValue(jsonStr, Test.class)); }}
复制代码


  1. 执行结果如下,可见 aaa、bbb 都被放入了 map 中:

Test{field0='null', map={aaa=value_aaa, field0=000, bbb=value_bbb}}
复制代码


  1. 另外 JsonAnySetter 还可以作用在成员变量上,上面的代码中,去掉 setValue 方法,在成员变量 map 上增加 JsonAnySetter 注解,修改后如下,执行结果也是一模一样的:

    static class Test {
private String field0;
@JsonAnySetter private Map<String, Object> map = new HashMap<>();
@Override public String toString() { return "Test{" + "field0='" + field0 + '\'' + ", map=" + map + '}'; } }
复制代码


  1. 注意,JsonAnySetter 作用在成员变量上时,该成员变量必须是 java.util.Map 的实现类

  • 至此,Jackson 常用注解已全部实战完毕,希望这些丰富的注解能助您制定出各种灵活的序列化和反序列化策略;

欢迎关注 InfoQ:程序员欣宸

学习路上,你不孤单,欣宸原创一路相伴...

发布于: 刚刚阅读数: 2
用户头像

搜索"程序员欣宸",一起畅游Java宇宙 2018.04.19 加入

前腾讯、前阿里员工,从事Java后台工作,对Docker和Kubernetes充满热爱,所有文章均为作者原创,个人Github:https://github.com/zq2599/blog_demos

评论

发布
暂无评论
jackson学习之八:常用方法注解_4月月更_程序员欣宸_InfoQ写作平台