gpt4 book ai didi

java - 在java中转换非结构化对象

转载 作者:行者123 更新时间:2023-12-03 11:17:28 30 4
gpt4 key购买 nike

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

最佳答案

我不这么认为(或者我不知道)Java 中的 MongoDB 是否通过某个字段提供这种动态转换(它需要指定哪些字段和哪些类)。但是你可以用手来做。
首先,您需要定义类型(枚举值或一些映射)以将字符串与类匹配。您可以创建抽象父类(例如 TypedObject )以便于使用和绑定(bind)所有目标类( VideoDocumentAssessment )。
接下来,您必须从 Mongo 读取值并将其映射到任何内容,因为您想读取代码中的所有数据。 Object很好,但我推荐Map<String, Object> (您的 Object 实际上就是那个 Map - 您可以通过调用 list.get(0).toString() 来检查它)。您也可以映射到StringDBObject或一些 JSON 对象 - 你必须阅读 "type"手动字段并从对象中获取所有数据。
最后,您可以将“数据包”(在我的示例中为 Map<String, Object>)转换为目标类。
现在您可以按目标类使用转换后的对象。为了证明这些实际上是目标类,我使用 toString 打印对象所有领域。
示例实现
类(class):

@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() .但无论如何我需要ObjectMappertoClazz转换“数据包”的方法。
  • 我用 Map<String, Object>而不仅仅是 Object .更复杂:List<Map<String, Object>> readObjects = objectMapper.readValue(json, new TypeReference<>() {}); .如果你愿意,你可以这样做:List<Object> readObjects2 = (List<Object>) objectMapper.readValue(json, new TypeReference<List<Object>>() {});

  • 结果:
    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瞄准你需要的类(class)(我建议在转换前检查 instance of)并使用:
    Video video = (Video) toClazz(readObjects.get(0), objectMapper);
    System.out.println(video.getVideoUrl());

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

    关于java - 在java中转换非结构化对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64296978/

    30 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com