欢迎访问我的GitHub

https://github.com/zq2599/blog_demos

内容:所有原创文章分类汇总及配套源码,涉及Java、Docker、Kubernetes、DevOPS等;

系列文章汇总

本篇概览

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

源码下载

  1. 如果您不想编码,可以在GitHub下载所有源码,地址和链接信息如下表所示(https://github.com/zq2599/blog_demos):
名称 链接 备注
项目主页 https://github.com/zq2599/blog_demos 该项目在GitHub上的主页
git仓库地址(https) https://github.com/zq2599/blog_demos.git 该项目源码的仓库地址,https协议
git仓库地址(ssh) git@github.com:zq2599/blog_demos.git 该项目源码的仓库地址,ssh协议
  1. 这个git项目中有多个文件夹,本章的应用在jacksondemo文件夹下,如下图红框所示:

  1. 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:



3. 测试代码和结果如下,可见反序列化时,是按照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注解:



5. 修改后的执行结果如下,原来的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常用注解已全部实战完毕,希望这些丰富的注解能助您制定出各种灵活的序列化和反序列化策略;

你不孤单,欣宸原创一路相伴

  1. Java系列
  2. Spring系列
  3. Docker系列
  4. kubernetes系列
  5. 数据库+中间件系列
  6. DevOps系列

欢迎关注公众号:程序员欣宸

微信搜索「程序员欣宸」,我是欣宸,期待与您一同畅游Java世界...

https://github.com/zq2599/blog_demos

jackson学习之八:常用方法注解的更多相关文章

  1. jackson学习之五:JsonInclude注解

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...

  2. jackson学习之七:常用Field注解

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...

  3. jackson学习之一:基本信息

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...

  4. jackson学习之二:jackson-core

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...

  5. jackson学习之三:常用API操作

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...

  6. jackson学习之四:WRAP_ROOT_VALUE(root对象)

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...

  7. jackson学习之九:springboot整合(配置文件)

    欢迎访问我的GitHub 这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos 系列文章汇总 jackson学习之一:基本信息 jac ...

  8. jackson学习之十(终篇):springboot整合(配置类)

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...

  9. springmvc学习笔记(常用注解)

    springmvc学习笔记(常用注解) 1. @Controller @Controller注解用于表示一个类的实例是页面控制器(后面都将称为控制器). 使用@Controller注解定义的控制器有如 ...

随机推荐

  1. 在Termux(非root的安卓Linux模拟器)中安装和使用ftp服务器(pure-ftpd)(原创)[简单极致]

    Termux是单用户的linux模拟器,所以应用的专属文件夹下面.usr/etc/下面没有passwd和group文件对多用户组配置,只在.usr/bin/下面有passwd二进制应用可以改变当前用户 ...

  2. 【Nginx学习笔记】-初识Nginx

    Nginx 目录 Nginx Nginx 特点 Nginx 基本功能 Nginx 使用场景 Nginx 安装/卸载 Docker 方式运行 Ubuntu上安装 卸载Nginx Nginx 命令 Ngi ...

  3. maven生命周期与插件

    目录 Maven生命周期 clean default site 命令与对应周期 插件与绑定 插件目标 插件绑定 内置绑定 自定义绑定 插件配置 本文主要是针对<maven实战>书中关键知识 ...

  4. ptgmk

    num >= 1125899906842624 ? "P" \ : num >= 1099511627776 ? "T" \ : num >= ...

  5. zabbix 监控tomcat

    zabbix 监控tomcat server端rpm -ivh jdk-8u20-linux-x64.rpmvi /etc/profileJAVA_HOME=/usr/java/jdk1.8.0_20 ...

  6. 【Oracle】10.2.0.1升级到10.2.0.5

    升级数据库到10.2.0.5   因是测试环境,不需要备份:如是生产系统,建议进行全备份后再进行升级操作,预防数据丢失造成不必要的影响.   步骤: 上传并解压补丁,安装前准备,安装补丁,预升级检查, ...

  7. 文件监控性能问题【BUG】

    文件监控性能问题[BUG] 背景:JAVA写了一个文件夹目录监控的程序,使用的是org.apache.commons.io.monitor 包,项目稳定运行了一个月,现场反馈,文件夹数据处理越来越慢, ...

  8. 《Go 语言并发之道》读后感 - 第四章

    <Go 语言并发之道>读后感-第四章 约束 约束可以减轻开发者的认知负担以便写出有更小临界区的并发代码.确保某一信息再并发过程中仅能被其中之一的进程进行访问.程序中通常存在两种可能的约束: ...

  9. python-列表包字典-根据字典的某一个键的值来进行排序

    python-列表包字典-根据字典的某一个键的值来进行排序 列表包字典的数据结构 要实现按照字典中的某一个键所对应的值进行排序 有两种办法 方法一,使用列表的sort方法 由小到大排 列表.sort( ...

  10. jackson学习之二:jackson-core

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...