【问题标题】:Converting the unstructured object in java在java中转换非结构化对象
【发布时间】:2021-01-25 12:49:56
【问题描述】:

我将 MongoDb 用于非结构化文档。当我进行聚合时,我将最终输出作为非结构化对象。为了方便起见,我发布了一些示例数据。实际对象有很多字段。 例如:

[
    { _id : "1", type: "VIDEO", videoUrl : "youtube.com/java"},
    { _id : "2", type: "DOCUMENT", documentUrl : "someurl.com/spring-boot-pdf"},
    { _id : "3", type: "ASSESSMENT", marks : 78}
]

上述对象类型对应的类是

@Data
public class Video{
    private String _id;
    private String type;
    private String videoUrl;
}

@Data
public class Document{
    private String _id;
    private String type;
    private String documentUrl;
}

@Data
public class Assessment{
    private String _id;
    private String type;
    private Integer marks;
}

由于我无法指定转换器类,我将所有对象作为Object.class 的列表,这是所有对象的通用类型。

List<Object> list = mongoTemplate.aggregate(aggregation, mongoTemplate.getCollectionName(YOUR_COLLECTION.class), Object.class).getMappedResults();

它正在工作,但是对于后端和前端开发人员来说这是不可读且不可维护的(例如:swagger ui)。所以我想出了一个解决方案,将所有字段都归为一个类。

@Data
@JsonInclude(JsonInclude.Include.NON_NULL) 
class MyConvetor{
    private String _id;
    private String type;
    private String videoUrl;
    private String documentUrl;
    private Integer marks;
}

这里Jackson 有助于忽略所有空字段

现在我可以使用MyConverter 作为类型

List<MyConverter> list = mongoTemplate.aggregate(aggregation, mongoTemplate.getCollectionName(YOUR_COLLECTION.class), MyConverter.class).getMappedResults();

但是当我们实现标准应用程序时,我觉得这不是一个好的做法。我想知道,有没有办法避免通用类型类(例如扩展任何抽象类)?或者这是我唯一能做的?

【问题讨论】:

  • 您的数据已经被标记为非结构化,在我看来,这是一个问题,比发送给您的客户的一般数据类型更严重。此外,我认为您的客户正在获得正确指定的数据结构,尤其是。 type 字段确定内容类型。您可以将@JsonInclude(JsonInclude.Include.NON_NULL) 视为序列化机制,即使它确实在上下文或需求中引入了代码异味。不过,您可能会更改的一件事是使 videoUrldocumentUrlmarks 成为组合类型的一部分,在 videoDetails 等字段中。
  • @ernest_k 这是有道理的。但我只是为了方便而发布了一个示例字段。那是我的错误,还没有包括在内。我的每个班级都有20多个领域。所以我不能做组合字段,不是吗?这就是问题

标签: java spring-boot spring-data spring-data-mongodb


【解决方案1】:

我不这么认为(或者我不知道)Java 中的 MongoDB 是否通过某些字段提供这种动态转换(它需要指定什么字段和什么类)。但是您可以手动完成。

首先,您需要定义类型(枚举值或一些映射)以将字符串与类匹配。您可以创建抽象父类(例如TypedObject)以便于使用和绑定所有目标类(VideoDocumentAssessment)。

接下来,您必须从 Mongo 读取值并将其映射到任何内容,因为您想读取代码中的所有数据。 Object 很好,但我推荐 Map&lt;String, Object&gt;(您的 Object 实际上是那个地图 - 您可以通过调用 list.get(0).toString() 来检查它)。您还可以映射到 StringDBObject 或一些 JSON 对象 - 您必须手动读取 "type" 字段并从对象中获取所有数据。

最后,您可以将“数据包”(在我的示例中为Map&lt;String, Object&gt;)转换为目标类。

现在您可以按目标类使用转换后的对象。为了证明这些实际上是目标类,我打印了带有 toString 所有字段的对象。

示例实现

类:

@Data
public abstract class TypedObject {
    private String _id;
    private String type;
}

@Data
@ToString(callSuper = true)
public class Video extends TypedObject {
    private String videoUrl;
}

@Data
@ToString(callSuper = true)
public class Document extends TypedObject {
    private String documentUrl;
}

@Data
@ToString(callSuper = true)
public class Assessment extends TypedObject {
    private Integer marks;
}

用于将字符串类型映射到类的枚举:

@RequiredArgsConstructor
public enum Type {
    VIDEO("VIDEO", Video.class),
    DOCUMENT("DOCUMENT", Document.class),
    ASSESSMENT("ASSESSMENT", Assessment.class);

    private final String typeName;
    private final Class<? extends TypedObject> clazz;

    public static Class<? extends TypedObject> getClazz(String typeName) {
        return Arrays.stream(values())
                .filter(type -> type.typeName.equals(typeName))
                .findFirst()
                .map(type -> type.clazz)
                .orElseThrow(IllegalArgumentException::new);
    }
}

将“数据包”从 JSON 转换为目标类的方法:

    private static TypedObject toClazz(Map<String, Object> objectMap, ObjectMapper objectMapper) {
        Class<? extends TypedObject> type = Type.getClazz(objectMap.get("type").toString());
        return objectMapper.convertValue(objectMap, type);
    }

将 JSON 读取到“数据包”并使用上述内容:

    String json = "[\n" +
            "    { _id : \"1\", type: \"VIDEO\", videoUrl : \"youtube.com/java\"},\n" +
            "    { _id : \"2\", type: \"DOCUMENT\", documentUrl : \"someurl.com/spring-boot-pdf\"},\n" +
            "    { _id : \"3\", type: \"ASSESSMENT\", marks : 78}\n" +
            "]";

    ObjectMapper objectMapper = new ObjectMapper();
    objectMapper.configure(JsonParser.Feature.ALLOW_UNQUOTED_FIELD_NAMES, true);

    List<Map<String, Object>> readObjects = objectMapper.readValue(json, new TypeReference<>() {});

    for (Map<String, Object> readObject : readObjects) {
        TypedObject convertedObject = toClazz(readObject, objectMapper);
        System.out.println(convertedObject);
    }

备注:

  • 在示例中,我使用 Jackson ObjectMapper 来读取 JSON。这使得示例和测试更简单。我认为您可以将其替换为mongoTemplate.aggregate()。但无论如何我需要toClazz 方法中的ObjectMapper 来转换“数据包”。
  • 我使用Map&lt;String, Object&gt; 而不仅仅是Object。它更复杂:List&lt;Map&lt;String, Object&gt;&gt; readObjects = objectMapper.readValue(json, new TypeReference&lt;&gt;() {});。如果你愿意,你可以这样做:List&lt;Object&gt; readObjects2 = (List&lt;Object&gt;) objectMapper.readValue(json, new TypeReference&lt;List&lt;Object&gt;&gt;() {});

结果:

Video(super=TypedObject(_id=1, type=VIDEO), videoUrl=youtube.com/java)
Document(super=TypedObject(_id=2, type=DOCUMENT), documentUrl=someurl.com/spring-boot-pdf)
Assessment(super=TypedObject(_id=3, type=ASSESSMENT), marks=78)

当然,您可以将TypedObject 转换为您需要的目标类(我建议在转换之前检查instance of)并使用:

Video video = (Video) toClazz(readObjects.get(0), objectMapper);
System.out.println(video.getVideoUrl());

我假设您阅读了整个集合一次,然后将所有类型混合在一个列表中(如您的问题中的示例)。但是您可以尝试通过字段"type" 在 MongoDB 中查找文档,并为每种类型分别获取数据。有了这个,您可以轻松地分别转换为每种类型。

【讨论】:

    猜你喜欢
    • 2015-01-05
    • 2021-02-05
    • 1970-01-01
    • 2021-02-14
    • 2020-04-20
    • 2017-11-26
    • 2020-10-03
    • 2021-07-17
    • 1970-01-01
    相关资源
    最近更新 更多