gpt4 book ai didi

spring - 在 1.4.5 中工作时升级到 Spring Boot 2.1.3 后阅读文档时如何让@TypeAlias 工作

转载 作者:行者123 更新时间:2023-12-05 03:02:13 31 4
gpt4 key购买 nike

我目前正在使用 spring data mongodb 并且配置文件扩展了 AbstractMongoConfiguration:

@Configuration
@EnableMongoRepositories(basePackages = "com.mycompany")
@EnableMongoAuditing
public class MongoConfig extends AbstractMongoConfiguration
{

我重写了 getMappingBasePackage() 方法来设置要像这样扫描的包:

    @Override
protected String getMappingBasePackage()
{
return "com.mycompany";
}

我一直在调试代码并注意到一些有趣的事情:

  1. 我在两个地方得到了 java.lang.InstantiationError。当我试图从 mongo 中读取引用了抽象类 (ParentClass) 的文档时,这两种情况都会发生。它试图实例化抽象类,而不是查找我已添加到子类的 @TypeAlias 注释。

这是我的父类的样子:

@Document
@JsonTypeInfo(use=JsonTypeInfo.Id.NAME, include=JsonTypeInfo.As.EXISTING_PROPERTY, visible=true, property="type")
@JsonSubTypes({
@Type(value=Child1.class, name="JSON_TYPE_CHILD1"),
@Type(value=Child2.class, name="JSON_TYPE_CHILD2"),
@Type(value=Child3.class, name="JSON_TYPE_CHILD3")
})
public abstract class ParentClass
{
...

我的 child 类(class)是这样的:

@Document
@JsonTypeName("JSON_TYPE_CHILD1")
@TypeAlias("ALIAS_TYPE_CHILD1")
public class Child1 extends ParentClass
{
...

这是我尝试读入的 json 的样子(经过简化):

{
"_id" : ObjectId("5c86d31388f13344f4098c64"),
"listOfWrapperClass" : [
{
"parentClass" : {
"type" : "JSON_TYPE_CHILD1",
"prop1" : 50.0,
"prop2" : 50.0,
"_class" : "ALIAS_TYPE_CHILD1"
},
"isReportOutOfDate" : false,

}
],
"_class" : "com.mycompany.domain.job.Job"
}

当我通过 spring 数据进行调试时,问题出现在 DefaultTypeMapper 中:

    private TypeInformation<?> getFromCacheOrCreate(Alias alias) {

Optional<TypeInformation<?>> typeInformation = typeCache.get(alias);

if (typeInformation == null) {
typeInformation = typeCache.computeIfAbsent(alias, getAlias);
}

return typeInformation.orElse(null);
}

它可以很好地加载包装器类,但是当它到达子类时,别名应该设置为“ALIAS_TYPE_CHILD1”,但以下值在 typeCache 中:

    {
NONE=Optional.empty,
ALIAS_TYPE_CHILD1=Optional.empty,
com.mycompany.domain.job.Job=Optional[com.mycompany.domain.job.Job]
}

因为键“ALIAS_TYPE_CHILD1”有一个 Optional.empty 作为值,代码没有获得正确的目标类型来加载,因此它使用 rawType,它是 ParentClass。因为它不能实例化一个抽象类而爆炸了。这是堆栈跟踪:

Caused by: java.lang.InstantiationError: com.mycompany.domain.job.base.ParentClass
at com.mycompany.domain.job.base.ParentClass_Instantiator_q3kytg.newInstance(Unknown Source)
at org.springframework.data.convert.ClassGeneratingEntityInstantiator$EntityInstantiatorAdapter.createInstance(ClassGeneratingEntityInstantiator.java:226)
at org.springframework.data.convert.ClassGeneratingEntityInstantiator.createInstance(ClassGeneratingEntityInstantiator.java:84)
at org.springframework.data.mongodb.core.convert.MappingMongoConverter.read(MappingMongoConverter.java:272)
at org.springframework.data.mongodb.core.convert.MappingMongoConverter.read(MappingMongoConverter.java:245)
at org.springframework.data.mongodb.core.convert.MappingMongoConverter.readValue(MappingMongoConverter.java:1491)
at org.springframework.data.mongodb.core.convert.MappingMongoConverter$MongoDbPropertyValueProvider.getPropertyValue(MappingMongoConverter.java:1389)
at org.springframework.data.mongodb.core.convert.MappingMongoConverter.readProperties(MappingMongoConverter.java:378)
at org.springframework.data.mongodb.core.convert.MappingMongoConverter.populateProperties(MappingMongoConverter.java:295)
at org.springframework.data.mongodb.core.convert.MappingMongoConverter.read(MappingMongoConverter.java:275)
at org.springframework.data.mongodb.core.convert.MappingMongoConverter.read(MappingMongoConverter.java:245)
at org.springframework.data.mongodb.core.convert.MappingMongoConverter.readCollectionOrArray(MappingMongoConverter.java:1038)
at org.springframework.data.mongodb.core.convert.MappingMongoConverter.readValue(MappingMongoConverter.java:1489)
at org.springframework.data.mongodb.core.convert.MappingMongoConverter$MongoDbPropertyValueProvider.getPropertyValue(MappingMongoConverter.java:1389)
at org.springframework.data.mongodb.core.convert.MappingMongoConverter.readProperties(MappingMongoConverter.java:378)
at org.springframework.data.mongodb.core.convert.MappingMongoConverter.populateProperties(MappingMongoConverter.java:295)
at ...

奇怪的是,如果我插入一个具有@TypeAlias("ALIAS_TYPE_CHILD1")首先的新文档,上面提到的typeCache是像这样正确填充:

{  
NONE=Optional.empty,
ALIAS_TYPE_CHILD1=Optional[com.mycompany.domain.job.base.Child1],
com.mycompany.domain.job.Job=Optional[com.mycompany.domain.job.Job]
}

当我在插入后立即执行 findOne 时,我可以毫无错误地读入文档,因为它使用 Child1 来实例化 pojo 而不是 ParentClass。如果我尝试先读取,那么我是否在之后插入都没有关系,因为 typeCace 在那里得到了错误的值,它会一直使用该值,直到您重新启动服务器。

我的猜测是配置或默认设置发生了变化。我能够解决所有其他升级问题,但这个让我感到困惑。如果 spring 数据中存在实际问题,我会感到震惊,因为我相信现在有人会遇到这个问题,因为我不可能是唯一一个尝试在 spring 中使用 @TypeAlias 的人-数据-mongodb。更不用说这一切都适用于我使用的以前版本的 spring boot(1.4.5 使用 spring-data-mongodb 1.9.8.RELEASE)。

欢迎就下一步尝试提出任何想法或建议。我完全不知道下一步该怎么做。

最佳答案

问题在于 typeCache 并没有在服务器启动时首先被填充。这是因为 protected String getMappingBasePackage()现在已弃用。你应该使用 protected Collection<String> getMappingBasePackages()相反,然后一切都很好。

重写这个方法解决了这个问题:

    @Override
protected Collection<String> getMappingBasePackages()
{
return Arrays.asList("com.mycompany");
}

关于spring - 在 1.4.5 中工作时升级到 Spring Boot 2.1.3 后阅读文档时如何让@TypeAlias 工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55114834/

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