多种类型的杰克逊反序列化接口

问题描述

我正在尝试在 Java 中使用 Jackson 反序列化遇到一些麻烦。我已经提出了 2 个解决方案,但我无法解决问题。问题?我得到了重复属性的结果,这是杰克逊反序列化后重复的字段。 (我的问题和这个问题完全一样:Avoid duplicate field generated by JsonTypeInfo in Jackson,当时没人能给你答案)

首先,我有以下课程:

@JsonIgnoreProperties(ignoreUnkNown = true)
public class Instance {

    @JsonProperty("id")
    private String id;
 
    @JsonProperty("name")
    private String name;

    @JsonProperty("type")
    private InstanceType type;
 }

我正在尝试做的只是实例化一个“实例”类型的对象,保存并读取它。使用解决方案 2,对象以重复的类型保存(类型显示为包含 'name'、'firs_type' 或 'second_type' 的数组)取决于我创建的内容。使用解决方案 1,我可以保存对象,但是当我尝试读取它时,我遇到了 jackson 异常转换。

解决方案 1:

@JsonDeserialize(using = InstanceTypeDeserializer.class)
public interface InstanceType {
    String value();
}

@JsonDeserialize(as = HardInstanceType.class)
public enum HardInstanceType implements InstanceType {
    FirsT_TYPE("first_type"),SECOND_TYPE("second_type")
    private String value;

    HardInstanceType(String value) {
        this.value = value;
    }

    @JsonValue
    public String value() {
       return value;
    }
}

@JsonDeserialize(as = SoftInstanceType.class)
public enum SoftInstanceType implements InstanceType {
    //.. types implementaion similar as HardInstanceType
}

public class InstanceTypeDeserializer extends JsonDeserializer<InstanceType> {
    @Override
    public InstanceType deserialize(JsonParser jp,DeserializationContext ctxt) throws IOException,JsonProcessingException {
        ObjectMapper mapper = (ObjectMapper) jp.getCodec();
        ObjectNode root = (ObjectNode) mapper.readTree(jp);
        
        if(root.get("name").asText().equals("hard")) {
            return mapper.readValue(root.toString(),HardInstanceType.class);
        } else { 
            return mapper.readValue(root.toString(),SoftInstanceType.class);
        }
    }
}

解决方案的问题在于,当我尝试存储数据并映射到类时,出现以下错误

异常解析json: com.fasterxml.jackson.databind.JsonMappingException:类 com.fasterxml.jackson.databind.node.TextNode 无法转换为类 com.fasterxml.jackson.databind.node.ObjectNode (com.fasterxml.jackson.databind.node.TextNode 和 com.fasterxml.jackson.databind.node.ObjectNode 在未命名的模块中 加载器 org.springframework.boot.loader.Launchedurlclassloader @1a3e8e24)(通过参考链: java.util.ArrayList[0]->com.project.package.xxxx.Instance["type"])

解决方案 2

@JsonTypeInfo(
    use = JsonTypeInfo.Id.NAME,include = JsonTypeInfo.As.PROPERTY,property = "name")
@JsonSubTypes({
    @JsonSubTypes.Type(value = HardInstanceType.class,name = "hard") })
public interface InstanceType {
   String value();
}

这个解决方案的问题是,当我保存数据时,当我创建一个实例对象并将其存储在数据存储中时,我得到以下内容

      "id": "1","name": "hard","type": [
        "hard","first_type"
      ]

什么是不正确的,类型应该只存储“first_type”(解决方案 1 存储了什么,但我看不懂哈哈)。

当然,Instace 类更复杂,字段也更多,我在这里简化一下,仅作为示例。

我需要这方面的帮助,非常感谢您。

解决方法

终于可以解决问题了。 我发布这个以防万一其他人需要它。

  1. 向我的 HardInstanceType 类添加一个属性。

    公共枚举 HardInstanceType 实现 InstanceType {

      FIRST_TYPE("first_type"),SECOND_TYPE("second_type");
      private String value;
    
      public String hardTypeIdentifierSer = "hardTypeIdentifierSer";
    
      HardInstanceType(String value) {
          this.value = value;
      }
    
      @JsonValue
      public String value() {
         return value;
      }
    

    }

  2. 然后,在解串器中:

    公共类 InstanceTypeDeserializer 扩展 JsonDeserializer {

     @Override
     public InstanceType deserialize(JsonParser jp,DeserializationContext ctxt) throws IOException,JsonProcessingException {
         TreeNode node = jp.readValueAsTree();
    
         if (node.get("hardTypeIdentifierSer") != null) {
             return jp.getCodec().treeToValue(node,HardInstanceType.class);
         }
    

    }