首页 > 解决方案 > 使用父属性值对杰克逊进行多态反序列化

问题描述

我有一个json如下:

{
    "transformationRules": [
        {
            "targetFieldName": "pageUrlNumber",
            "inputPath": "$.metadata.pageUrl",
            "rules": [
                {
                    "name": "regex",
                    "regexPattern": "\\d+",
                    "regexTransformType": "extract",
                    "order":2
                }
            ]
        },
        {
            "targetFieldName": "categories",
            "inputPath": "$.attributes[0].productInfo.breadCrumbs",
            "rules": [
                {
                    "name": "transform"
                }
            ]
        }
    ]
}

rules数组具有不同类的json。

我想根据名称的值将上面的 json 转换为他们的 pojo,即 regex,transform。我已经尝试了以下代码。

规则.java

@NoArgsConstructor
@Data
@JsonTypeInfo(
        use = JsonTypeInfo.Id.NAME,
        include = JsonTypeInfo.As.EXTERNAL_PROPERTY,
        property = "name")
@JsonSubTypes({
        @JsonSubTypes.Type(value = RegexRule.class, name = "regex"),
        @JsonSubTypes.Type(value = Transform.class, name = "transform"),
})
public abstract class Rule {

    @JsonProperty("name")
    private String name;

    @JsonProperty("order")
    private int order = 1;

    protected Rule(String name, int order) {
        this.name = name;
        this.order = order;
    }

}

正则表达式规则.java

@NoArgsConstructor
@Data
public class RegexRule extends Rule{

    @JsonProperty("regexPattern")
    private String regexPattern;

    @JsonProperty("regexTransformType")
    private String regexTransformType;

    @JsonProperty("replacementValue")
    private String replacementValue = "";

    @JsonProperty("regexGroupNumber")
    private int regexGroupNumber = 0;

    public RegexRule(String name, int order, String regexPattern, String regexTransformType, String replacementValue, int regexGroupNumber) {
        super(name, order);
        this.regexGroupNumber = regexGroupNumber;
        this.regexPattern = regexPattern;
        this.regexTransformType = regexTransformType;
        this.replacementValue = replacementValue;
    }
}

转换.java

@NoArgsConstructor
@Data
public class Transform extends Rule{

    public Transform(String name, int order) {
        super(name, order);
    }
}

主.java

for(TransformationRule transformationRule: getTransformationRules()){
            for(Rule rule:transformationRule.getRules()){
                System.out.println(rule.getName());
            }
        }

输出:

null
null

在我的 Main 函数中,当我将 json 反序列化为 pojos 时,我将 name 设为 null。相反,我应该得到regexandtransform作为输出

我在这里犯了什么错误?我需要工厂方法的名称

标签: javajsonjacksonjackson-databindjackson2

解决方案


As.EXTERNAL_PROPERTY在这里听起来是错误的,我很惊讶如果这能奏效,那就意味着:

{
  "name": "regex", // This is the *external* property, one level up
  { // this is the Rule
    "order": 2,
    ...
  }
}

尝试

  • As.PROPERTY
    • 您可以从 POJO 中删除Rule#name,因为它被视为仅存在于Jackson 正确反序列化的元数据。记住rule.getName()=="regex"是一样的rule instanceof RegexRule
  • As.EXISTING_PROPERTY
    • 将按照您的预期填充该字段,但在规模上可能会有稍差的性能(仅当这是应用程序持续执行的操作时才重要)
    • 有关另一个示例,请参见原始功能请求

推荐阅读