反序列化器的三种实现

本文介绍了一种在Java项目中处理第三方接口返回的复杂数据类型的解决方案,通过三种方法实现反序列化,包括使用注解、自定义反序列化器及在ObjectMapper中注册Module。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

0 背景

项目中调用第三方接口,返回的接口中包含多种数据类型,因此用反序列化器实现返回数据到具体类型的转换。本文提供三种实现方法(本地均已测试通过)。

1、以下面类作为示例

Item为返回数据的结构,其中data属性是指不同的返回数据类型,用泛型T表示:

public class Item<T> {
    private String name;
    private String type;
    private T data;     //根据type类型,将data反序列化为Data或者Model
}

DataItem和ModelItem实现如下,其中Data和Model是返回数据包含的两种类,这里不作说明:

public class DataItem extends Item<Data>{
}
public class ModelItem extends Item<Model>{
}

2、三种方法实现反序列化过程

方法一:注解(推荐)

              直接在类上加注解

import com.fasterxml.jackson.annotation.JsonSubTypes;
import com.fasterxml.jackson.annotation.JsonTypeInfo;

@JsonTypeInfo(
    use = JsonTypeInfo.Id.NAME,
    include = JsonTypeInfo.As.PROPERTY,
    property = "type"         //type类型
)
@JsonSubTypes(
    {
        @JsonSubTypes.Type(name = "model",value = ModelItem.class),  //根据type类型,反序列化到不同类
        @JsonSubTypes.Type(name = "data" ,value = DataItem.class)
    }
)
public class Item<T> {
    private String name;
    private String type;
    private T data;     //根据type类型,将data反序列化为Data或者Model
}

 

@JsonTypeName("model")     //这里对应Item类里type的类型
public class ModelItem extends Item<Model>{
}

 

@JsonTypeName("data")     //这里对应Item类里type的类型
public class DataItem extends Item<Data>{
}

方法二:反序列化器+注解

              定义反序列化器JsonItemDeserializer,同时在Item上加注解

public class JsonItemDeserializer extends JsonDeserializer<Item> {
    @Override
    public Item deserialize(JsonParser p, DeserializationContext ctxt) throws IOException, JsonProcessingException {
        ObjectCodec codec = p.getCodec();
        JsonNode treeNode = codec.readTree(p);
        String type = treeNode.get("type").asText();
        if ("data".equals(type)){
            return codec.treeToValue(treeNode, DataItem.class);
        }else if ("model".equals(type)){
            return codec.treeToValue(treeNode, ModelItem.class);
        }else {
            return null;
        }
    }
}
@JsonDeserialize(using = JsonItemDeserializer.class)
public class Item<T> {
    private String name;
    private String type;
    private T data;
}

方法三:反序列化器+ObjectMapper注册Module

              反序列化器JsonItemDeserializer同上

public class JsonItemDeserializer extends JsonDeserializer<Item> {
    @Override
    public Item deserialize(JsonParser p, DeserializationContext ctxt) throws IOException, JsonProcessingException {
        ObjectCodec codec = p.getCodec();
        JsonNode treeNode = codec.readTree(p);
        String type = treeNode.get("type").asText();
        if ("data".equals(type)){
            return codec.treeToValue(treeNode, DataItem.class);
        }else if ("model".equals(type)){
            return codec.treeToValue(treeNode, ModelItem.class);
        }else {
            return null;
        }
    }
}

         定义WebConfig类,定义init方法,并添加@PostConstruct注解,在ObjectMapper注册Module

@Configuration
public class WebConfig {
    private final ObjectMapper objectMapper;

    public WebConfig(ObjectMapper objectMapper) {
        this.objectMapper = objectMapper;
    }

    @PostConstruct
    public void init(){
        // 在ObjectMapper注册Module
        SimpleModule module = new SimpleModule();
        module.addDeserializer(Item.class, new JsonItemDeserializer());
        objectMapper.registerModule(module);
    }
    @Bean
    public RestTemplate restTemplate(){
        RestTemplate restTemplate = new RestTemplate();
        List<HttpMessageConverter<?>> converters = restTemplate.getMessageConverters();
        converters.removeIf(httpMessageConverter -> httpMessageConverter instanceof MappingJackson2HttpMessageConverter);
        // 删除原json转换,添加注册过的objectMapper
        converters.add(new MappingJackson2HttpMessageConverter(objectMapper));
        return restTemplate;
    }
}

3 使用反序列化后的类

        for (Item item : items){
            if (item.getData() instanceof Data){
                Data data = (Data)item.getData();
            }
            if (item.getData() instanceof Model){
                Model model = (Model)item.getData();
            }
        }

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值