前言

Github:https://github.com/HealerJean

博客:http://blog.healerjean.com

Jackson允许配置多态类型处理,当进行反序列话时,Json数据匹配的对象可能有多个子类型,为了正确的读取对象的类型,我们需要添加一些类型信息。

@JsonTypeInfo(
  use = JsonTypeInfo.Id.NAME,
  include = JsonTypeInfo.As.EXTERNAL_PROPERTY,
  property = "transType"
)
@JsonSubTypes({
  @JsonSubTypes.Type(value = LoanTrialDTO.class, name = "T00002"),
})
public void setServiceData(ServiceBaseDTO serviceData) {
  this.serviceData = serviceData;
}

1、解释

1.1、@JsonTypeInfo

@JsonTypeInfo 这个注解可以直接放在类上,也可以放在某个属性上:下面是其的属性值

1.1.1、use: (必选):

use: (必选):定义使用哪一种类型识别码(property 为识别码的 key),可选值有多种:在序列化时标志出不同的类型用什么区分,用在反序列化时转换成响应的类型

use属性值 若不指定property则默认 作用 是否依赖 JsonTypeInfo 的值
JsonTypeInfo.Id.NAME @type 使用 JsonTypeInfo的值作为识别码的值 如果有多个子类的情况,必须有 @JsonSubTypes,否则无法判断是哪个子类
JsonTypeInfo.Id.CLASS @class 用类的全路劲名称来作为识别码的值 与是否有@JsonSubTypes无关
JsonTypeInfo.Id.MINIMAL_CLASS @c 表示具有最小路径的Java类名称用作识别 是否有@JsonSubTypes无关
JsonTypeInfo.Id.NONE 暂不介绍    
JsonTypeInfo.Id.CUSTOM 暂不介绍    

1.1.2、include (可选)

设置识别码包含在哪里。 包含类型元数据的一种机制

include属性值 作用    
JsonTypeInfo.As.PROPERTY 作为 POJO 的属性出现 默认  
JsonTypeInfo.As.WRAPPER_OBJECT 作为一个包装的对象    
JsonTypeInfo.As.WRAPPER_ARRAY 作为一个包装的数组    
JsonTypeInfo.As.EXTERNAL_PROPERTY 作为一个额外的属性,跟 POJO 同级,只能用于属性,如何作用于类则跟JsonTypeInfo.As.PROPERTY 是相同效果    
JsonTypeInfo.As.EXISTING_PROPERTY 序列化,则 Jackson 不主动处理,由我们自行处理。 反序列化的时候,跟JsonTypeInfo.As.PROPERTY 的处理相同;    

1.1.3、property 可选)

设置识别码是名称, 自定义的区分类型的 id ,根据 use 的属性值不同,默认值不同,具体默认值看(1.1.1)

1.1.4、visible (可选)

visible(可选):定义识别码在反序列化时是否保留(不管false或true都不影响序列化)。默认是false,表示Jackson可以将识别码识别为类型后就删除。

1.2、@JsonSubTypes

可以用来表明这个父类的子类型有哪些

2、准备

2.1、Demo实体类

public abstract class Human {

    private String district;

    @Data
    public static class Man extends Human {
        private String manField;
    }

    @Data
    public static class Woman extends Human {
        private String womanField;
    }

}

2.2、正常情况下的错误的演示



    @Test
    public void normal() throws IOException {
        ObjectMapper mapper = new ObjectMapper();
        Man man = new Man();
        man.setManField("男人");
        man.setDistrict("山西");
        String json = mapper.writeValueAsString(man);
        System.out.println(json);
        // {"district":"山西","manField":"男人"}
        //报错 子类转父类,再不能直接序列化为子类
        man =((Man)mapper.readValue(json, Human.class)) ;
    }
    

3、开始

3.1、实体demo

@Data
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME,
        include = JsonTypeInfo.As.PROPERTY
)
@JsonSubTypes({
        @JsonSubTypes.Type(value = Human.Man.class, name = "man"),
        @JsonSubTypes.Type(value = Human.Woman.class, name = "woman"),
})
public abstract class Human {

    private String district;

    @Data
    public static class Man extends Human {
        private String manField;
    }

    @Data
    public static class Woman extends Human {
        private String womanField;
    }

}

3.2、测试用例

  @Test
    public void testOne() throws IOException {
        ObjectMapper mapper = new ObjectMapper();
        Man man = new Man();
        man.setManField("男人");
        man.setDistrict("北京");

        String manJson = mapper.writeValueAsString(man);
        log.info("序列化Man :【 {} 】", manJson);
        Human human = mapper.readValue(manJson, Human.class);
        log.info("子类转父类 ======================");
        String humanJson = mapper.writeValueAsString(human);
        log.info("反序列化man -> Human :【 {} 】", humanJson);
        log.info("human.getDistrict()  :【 {} 】",  human.getDistrict());
}

3.1.3、Use (必选)

3.1.3.1、use = JsonTypeInfo. Id.NAME

使用 JsonTypeInfo的值作为识别码的值,如果有多个子类的情况,必须有 @JsonSubTypes,否则无法判断是哪个子类

@JsonTypeInfo(use = JsonTypeInfo.Id.NAME)
@JsonSubTypes({
  @JsonSubTypes.Type(value = Human.Man.class, name = "man"),
  @JsonSubTypes.Type(value = Human.Woman.class, name = "woman"),
})


序列化Man :【 {"@type":"man","district":"北京","manField":"男人"} 
  子类转父类 ======================
  反序列化man -> Human :【 {"@type":"man","district":"北京","manField":"男人"} 
  human.getDistrict()  :【 北京 
  • **use = JsonTypeInfo.Id.NAME, 如果不写property ,则识别码 的名字默认为 @type **

  • 根据Json数据,识别之后,其值为@JsonSubTypes.Type(value = Human.Man.class, name = "man")中子类Man的name值

1、配置 property = “type”

@JsonTypeInfo(use = JsonTypeInfo.Id.NAME
			  property = "type"
)
@JsonSubTypes({
        @JsonSubTypes.Type(value = Human.Man.class, name = "man"),
        @JsonSubTypes.Type(value = Human.Woman.class, name = "woman"),
})

序列化Man :【 {"type":"man","district":"北京","manField":"男人"}  
子类转父类 ======================
反序列化man -> Human :【 {"type":"man","district":"北京","manField":"男人"} 
human.getDistrict()  :【 北京  
  • 这个时候默认”@type就变成了type,如果换成其他的use,则默认值失效,变成这里的property 的值

3.1.3.2、use = JsonTypeInfo. Id.CLASS、property = “type”

**用类的全路劲名称来作为识别码的值,和@JsonSubType没关系,写上了也不起作用,可以自动判断有哪些子类 **

@JsonTypeInfo(use = JsonTypeInfo.Id.CLASS,
        	  property = "type"
)
@JsonSubTypes({
        @JsonSubTypes.Type(value = Human.Man.class, name = "man"),
        @JsonSubTypes.Type(value = Human.Woman.class, name = "woman"),
})

序列化Man :【 {"type":"com.healerjean.proj.a_test.json.jackson.d02_JsonType.Human$Man","district":"北京","manField":"男人"}  
反序列化man -> Human :【 {"type":"com.healerjean.proj.a_test.json.jackson.d02_JsonType.Human$Man","district":"北京","manField":"男人"}  
human.getDistrict()  :【 北京  
  • **use = JsonTypeInfo.Id.CLASS, 如果不写property ,则识别码 的名字默认为 @class **

  • 根据Json数据,识别之后,值为,这个子类所在类路径,用.连接

1、这个时候发现和@JsonSubTypes中的值似乎没有什么关系,我果断去掉@JsonSubTypes属性
@JsonTypeInfo(use = JsonTypeInfo.Id.CLASS)


序列化Man :【 {"@class` ":"com.healerjean.proj.a_test.json.jackson.d02_JsonType.Human$Man","district":"北京","manField":"男人"}  
反序列化man -> Human :【 {"@class` ":"com.healerjean.proj.a_test.json.jackson.d02_JsonType.Human$Man","district":"北京","manField":"男人"}  
human.getDistrict()  :【 北京  

3.1.3.3、use = JsonTypeInfo. Id.MINIMAL_CLASS

**表示具有最小路径的Java类名称用作识别码的值,和@JsonSubType没关系,写上了也不起作用,可以自动判断有哪些子类 **

@Data
@JsonTypeInfo(use = JsonTypeInfo.Id.MINIMAL_CLASS)


序列化Man :【 {"@c":".Human$Man","district":"北京","manField":"男人"} 
子类转父类 ====================== 
反序列化man -> Human :【 {"@c":".Human$Man","district":"北京","manField":"男人"} human.getDistrict()  :【 北京 

  • **use = JsonTypeInfo.Id.CLASS, 如果不写property ,则识别码 的名字默认为 @c **

3.1.4、include (可选)

默认是 JsonTypeInfo.As.PROPERTY,上面所有的都是以它进行测试的

3.1.4.1、include = JsonTypeInfo.As.WRAPPER_OBJECT

作为一个包装的对象

@Data
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME,
        include = JsonTypeInfo.As.WRAPPER_OBJECT,
        property = "type"
)
@JsonSubTypes({
        @JsonSubTypes.Type(value = Human.Man.class, name = "man"),
        @JsonSubTypes.Type(value = Human.Woman.class, name = "woman"),
})


序列化Man :【 {"man":{"district":"北京","manField":"男人"}} 】
子类转父类 ====================== 
反序列化man -> Human :【 {"man":{"district":"北京","manField":"男人"}} 】
human.getDistrict()  :【 北京 】 

3.1.4.2、include = JsonTypeInfo.As.WRAPPER_ARRAY

作为一个包装的数组看下面,这个数组可是不太规则哦,就是说里面放的不一定的同类型的),和有无property无关,看下面的测试打印结果就知道

@JsonTypeInfo(use = JsonTypeInfo.Id.NAME,
        include = JsonTypeInfo.As.WRAPPER_ARRAY,
        property = "type"
)
@JsonSubTypes({
        @JsonSubTypes.Type(value = Human.Man.class, name = "man"),
        @JsonSubTypes.Type(value = Human.Woman.class, name = "woman"),
})


序列化Man :【 ["man",{"district":"北京","manField":"男人"}] 
子类转父类 ====================== 
反序列化man -> Human :【 ["man",{"district":"北京","manField":"男人"}] 
human.getDistrict()  :【 北京 

3.1.4.3、include = JsonTypeInfo.As.EXTERNAL_PROPERTY

作为一个额外的属性,跟POJO同级,只能用于属性,如何作用于类则跟JsonTypeInfo.As.PROPERTY是相同效果

@Data
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME,
        include = JsonTypeInfo.As.EXTERNAL_PROPERTY,
        property = "type"
)
@JsonSubTypes({
        @JsonSubTypes.Type(value = Human.Man.class, name = "man"),
        @JsonSubTypes.Type(value = Human.Woman.class, name = "woman"),
})


序列化Man :【 {"type":"man","district":"北京","manField":"男人"} 
类转父类 ======================
反序列化man -> Human :【 {"type":"man","district":"北京","manField":"男人"} 
human.getDistrict()  :【 北京 

3.1.4.4、include = JsonTypeInfo.As.EXISTING_PROPERTY

反序列化 :

  • 这个属性可是骚的厉害了,它要求property识别码的名称必须在被该属性表示的类中也存在一份,否则反序列化就会报错。
  • 反序列化的时候这个识别码名字属性被认为是识别码,如果这个识别码的值,不在@JsonSubTypes 的name中就会报错
  • 再者,反序列化之后这个值必须和我们真正意义上的反序列化的子类的JsonSubTypes 中 name一样,不可以任意挑选,否则会报错

序列化 :

  • 跟它没半毛钱关系,和整个注解也没关系

序列化的时候,跟它没半毛钱关系,但是不会像上面的那样

@Data
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME,
        include = JsonTypeInfo.As.EXISTING_PROPERTY,
        property = "type"
)
@JsonSubTypes({
        @JsonSubTypes.Type(value = Human.Man.class, name = "man"),
        @JsonSubTypes.Type(value = Human.Woman.class, name = "woman"),
})
public class Human {

    private String district;

    private String type;

    @Data
    public static class Man extends Human {
        private String manField;
    }

    @Data
    public static class Woman extends Human {
        private String womanField;
    }

}




序列化Man :【 {"district":"北京","type":"man","manField":"男人"} 
子类转父类 ======================
 反序列化man -> Human :【 {"district":"北京","type":null,"manField":"男人"} 
 human.getDistrict()  :【 北京 
 

2.1.5、property (可选)

设置识别码是名称,根据use的属性,一般情况下有默认值

除了JsonTypeInfo.As.EXISTING_PROPERTY 有些特殊之外,JsonTypeInfo.As.EXTERNAL_PROPERTY、JsonTypeInfo.As.PROPERTY 注解的类中有了相同的属性(剩下的不讲解,因为其他的和property 没半毛钱关系),则会出现两次 以JsonTypeInfo.As.PROPERTY 举例

注意点

  • 在visible 为默认值 false的情况下反序列化, 虽然会出现两次,但是在位置上有严格要求,我测试过好多次,发现这个标识码肯定是靠前的,而且不能删除,删除就错了(当然啊,删除了,怎么判断你是哪个子类)

@Data
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME,
        include = JsonTypeInfo.As.EXTERNAL_PROPERTY,
        property = "type"
)
@JsonSubTypes({
        @JsonSubTypes.Type(value = Human.Man.class, name = "man"),
        @JsonSubTypes.Type(value = Human.Woman.class, name = "woman"),
})
public class Human {

    private String district;

    private String type;

    @Data
    public static class Man extends Human {
        private String manField;
    }

    @Data
    public static class Woman extends Human {
        private String womanField;
    }

}



   @Test
    public void testOne() throws IOException {
        ObjectMapper mapper = new ObjectMapper();
        Man man = new Man();
        man.setManField("男人");
        man.setDistrict("北京");
        man.setType("lalala");

        String manJson = mapper.writeValueAsString(man);
        log.info("序列化Man :【 {} 】", manJson);
        Human human = mapper.readValue(manJson, Human.class);
        log.info("子类转父类 ======================");
        String humanJson = mapper.writeValueAsString(human);
        log.info("反序列化man -> Human :【 {} 】", humanJson);
        log.info("human.getDistrict()  :【 {} 】",  human.getDistrict());
        log.info("human.getType()  :【 {} 】",  human.getType());
    }



序列化Man :【 {"type":"man","district":"北京","type":"lalala","manField":"男人"} 
子类转父类 ======================
反序列化man -> Human :【 {"type":"man","district":"北京","type":"lalala","manField":"男人"}  
human.getDistrict()  :【 北京 
 human.getType()  :【 lalala  
  

2.1.6、visible(可选)

visible(可选):定义识别码在反序列化(反序列为对象内部是否保留给字段)时是否保留(不管false或true都不影响序列化)。默认是false,表示Jackson可以将识别码识别为类型后就删除。

 @Test
    public void testOne() throws IOException {
        ObjectMapper mapper = new ObjectMapper();
        Man man = new Man();
        man.setManField("男人");
        man.setDistrict("北京");
        man.setType("1");

        String manJson = mapper.writeValueAsString(man);
        log.info("序列化Man :【 {} 】", manJson);
        //只剩下一个type,必须放在第一个,这样才能识别子类
        manJson = "{\"type\":\"man\",\"district\":\"北京\",\"manField\":\"男人\"} ";
        Human human = mapper.readValue(manJson, Human.class);
        log.info("子类转父类 ======================");
        String humanJson = mapper.writeValueAsString(human);
        log.info("反序列化man -> Human :【 {} 】", humanJson);
        log.info("human.getDistrict()  :【 {} 】",  human.getDistrict());
        log.info("human.getType()  :【 {} 】",  human.getType());
    }
    
    
    
  • 一般情况下 visible = false
manJson = "{\"type\":\"man\",\"district\":\"北京\",\"manField\":\"男人\"} ";

@Data
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME,
        include = JsonTypeInfo.As.PROPERTY,
        property = "type",
        visible = false
)
@JsonSubTypes({
        @JsonSubTypes.Type(value = Human.Man.class, name = "man"),
        @JsonSubTypes.Type(value = Human.Woman.class, name = "woman"),
})


序列化Man :【 {"type":"man","district":"北京","type":"1","manField":"男人"} 
子类转父类 ====================== 
反序列化man -> Human :【 {"type":"man","district":"北京","type":null,"manField":"男人"} 
human.getDistrict()  :【 北京 
human.getType()  :【 null 




  • visible = true

反序列化会把它当做一个属性值处理(保留给字段了),也就是说反序列化后还是可见的,如果在这个上面我测试的Json中再加一个type,那么这个这个识别码照样能打印,但是会失效

序列化Man :【 {"type":"man","district":"北京","type":"1","manField":"男人"} 】
子类转父类 ======================
反序列化man -> Human :【 {"type":"man","district":"北京","type":"man","manField":"男人"} 】
 human.getDistrict()  :【 北京 】
human.getType()  :【 man 】



如果在这个上面我测试的Json中再加一个type,那么这个这个识别码照样能打印,但是会失效
manJson = "{\"type\":\"man\",\"district\":\"北京\",\"type\":\"1\",\"manField\":\"男人\"} ";

打印结果
反序列化man -> Human :【 {"type":"man","district":"北京","type":"1","manField":"男人"} 】
human.getType()  :【 1 】

ContactAuthor