java - 将 JSON 映射到具有单属性和多属性的多态 POJO

标签 java json annotations jackson

我目前正在尝试开发一个与特定在线服务通信的 REST 客户端。此在线服务返回一些 JSON 响应,我希望使用 Jackson 将其映射到 Java 对象。

一个 JSON 响应的例子是:

{
    "id" : 1,
    "fields" : [ {
      "type" : "anniversary",
      "value" : {
        "day" : 1,
        "month" : 1,
        "year" : 1970
      }
    }, {
      "type" : "birthday",
      "value" : {
        "day" : 1,
        "month" : 1,
        "year" : 1970
      }
    }, {
      "type" : "simple",
      "value" : "simple string"
    },{
       "type": "name",
       "value": {
           "firstName": "Joe",
           "lastName": "Brown"
       }
    } ]
}

注意以下几点:

  • 这个结构包含一个简单的 id 和 Field 实例的集合,每个实例都有一个类型和一个
  • value 结构由外部属性 type 决定
  • 在给定的例子中,有3种类型->日期、姓名和单值字符串
  • birthdayanniversary 类型都匹配date 结构
  • 有几种映射到单个值字符串的类型,如 email 类型、twitterId 类型、company 类型等。

My problem is that I do not seem to be able to correctly map this structure to the Java objects

这是我到目前为止所做的。以下是类及其 Jackson 注释(省略了 getter 和 setter)。

public class Contact {
    private int id;
    private List<Field> fields;
}

public class Field {
    private FieldType type;
    private FieldValue value;
}

public enum FieldType {
    EMAIL, NICKNAME, NAME, ADDRESS, BIRTHDAY, ANNIVERSARY
}

@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = JsonTypeInfo.As.EXTERNAL_PROPERTY, property = "type",
        defaultImpl = SingleFieldValue.class)
@JsonSubTypes({
        @JsonSubTypes.Type(value = NameFieldValue.class, name = "name"),
        @JsonSubTypes.Type(value = DateFieldValue.class, name = "anniversary"),
        @JsonSubTypes.Type(value = DateFieldValue.class, name = "birthday"),
        @JsonSubTypes.Type(value = SingleFieldValue.class, name = "nickname"),
        @JsonSubTypes.Type(value = SingleFieldValue.class, name = "email"),
        //other types that map to SingleFieldValue
})
public abstract FieldValue {
}


public class NameFieldValue extends FieldValue {

    private String firstName;
    private String lastName;
 }

public class DateFieldValue extends FieldValue {

    private int day;
    private int month;
    private int year;
}

public class SingleFieldValue extends FieldValue {

    private String value;
}

ObjectMapper不包含任何配置,使用默认配置。

您对正确映射这些有什么建议?我想避免制作自定义反序列化器,而只是遍历 Json 对象,例如 JsonNode。

注意:对于缺乏足够的信息来明确说明这个问题,我提前表示歉意。请说明我的公式有任何问题。

最佳答案

您已经在 FieldValue 级别上使用了一个抽象类,以便在 FIeld 类中使用它。在这种情况下,您可以使用 type=email 和 value=address 构造对象,这可能会导致一些问题...

我建议为每个具有特定 FieldValue 类型的类型创建一个特定类。 以下代码将 JSON 序列化/反序列化为来自/到 POJO 的所需格式:

public class Main {
    String json = "{\"id\":1,\"fields\":[{\"type\":\"SIMPLE\",\"value\":\"Simple Value\"},{\"type\":\"NAME\",\"value\":{\"firstName\":\"first name\",\"lastName\":\"last name\"}}]}";

    public static void main(String []args) throws IOException {
        ObjectMapper objectMapper = new ObjectMapper();

        String json = objectMapper.writeValueAsString(generate());
        System.out.println(json);

        System.out.println(objectMapper.readValue(json, Contact.class));
    }

    private static Contact generate() {
        SimpleField simpleField = SimpleField.builder().type(FieldType.SIMPLE).value("Simple Value").build();

        NameFieldValue nameFieldValue = NameFieldValue.builder().firstName("first name").lastName("last name").build();
        NameField nameField = NameField.builder().type(FieldType.NAME).value(nameFieldValue).build();

        return Contact.builder().id(1).fields(Arrays.asList(simpleField, nameField)).build();
    }
}

@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = JsonTypeInfo.As.EXTERNAL_PROPERTY, property = "type")
@JsonSubTypes({
        @JsonSubTypes.Type(value = SimpleField.class, name = "SIMPLE"),
        @JsonSubTypes.Type(value = NameField.class, name = "NAME")
})
interface Field {
    FieldType getType();
    Object getValue();
}

enum FieldType {
    SIMPLE, NAME
}

@Data
@Builder
@AllArgsConstructor
@NoArgsConstructor
class Contact {
    private int id;
    private List<Field> fields;
}

@Data
@Builder
@AllArgsConstructor
@NoArgsConstructor
class SimpleField implements Field {
    private FieldType type;
    private String value;

    @Override
    public FieldType getType() {
        return this.type;
    }

    @Override
    public String getValue() {
        return this.value;
    }
}

@Data
@Builder
@AllArgsConstructor
@NoArgsConstructor
class NameField implements Field {
    private FieldType type;
    private NameFieldValue value;

    @Override
    public FieldType getType() {
        return this.type;
    }

    @Override
    public Object getValue() {
        return this.value;
    }
}

@Data
@Builder
@AllArgsConstructor
@NoArgsConstructor
class NameFieldValue {
    private String firstName;
    private String lastName;
}

我在这里使用 lombok 库只是为了最小化代码并避免创建 getter/setter 以及构造函数。您可以删除 lombok 注释并添加 getters/setters/constructors,代码将同样工作。

因此,我们的想法是您有一个 Contact 类(它是您的 JSON 的根)和一个 Fields 列表(其中 Field 是一个接口(interface))。每个 Field 类型都有自己的实现,例如 NameField 实现 Field 并将 NameFieldValue 作为属性。这里的技巧是您可以更改 getValue() 方法声明并声明它返回通用接口(interface)或对象(我使用了对象,但接口(interface)也可以)。

此解决方案不需要任何自定义序列化器/反序列化器,并且易于维护。

关于java - 将 JSON 映射到具有单属性和多属性的多态 POJO,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27271179/

相关文章:

java - 如何使用 Java 在 mongodb 中上传 json 文件?

mysql - 使用 JPA 注释设置 Long 值的列长度

java - Flink状态变量和普通类变量的区别

java - 使用 Liberty 配置文件配置 log4jdbc-log4j2

java - 有没有办法覆盖 Android Room 实体构造函数?

java - 注释在 Java 或任何其他编程语言中如何在内部工作?

java - 使用@Aspect更改方法签名

java - Maven 不接受 Spring Boot 应用程序中的测试

java - Jackson JSON解析映射转换

javascript - 使用 D3 访问嵌套的 JSON 数据