如何在升级到 Spring Boot 2.1.3 后阅读文档时在 1.4.5 中工作时使 @TypeAlias 工作
How to get @TypeAlias working when reading document after upgrading to Spring Boot 2.1.3 when it was working in 1.4.5
我目前正在使用 spring 数据 mongodb 并且配置文件扩展了 AbstractMongoConfiguration:
@Configuration
@EnableMongoRepositories(basePackages = "com.mycompany")
@EnableMongoAuditing
public class MongoConfig extends AbstractMongoConfiguration
{
我重写 getMappingBasePackage()
方法来设置包扫描如下:
@Override
protected String getMappingBasePackage()
{
return "com.mycompany";
}
我一直在调试代码并注意到一些有趣的事情:
- 我在两个地方得到
java.lang.InstantiationError
。当我试图从 mongo 中读取引用了抽象 class (ParentClass) 的文档时,这两种情况都会发生。它试图实例化抽象 class 而不是找到我添加到 child classes. 的 @TypeAlias
注释
这是我的 ParentClass 的样子:
@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);
}
它加载包装器 class 很好,但是当它到达 child class 时,别名设置为 "ALIAS_TYPE_CHILD1",但以下值在类型缓存:
{
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。因为它不能实例化一个抽象 class 而爆炸了。这是堆栈跟踪:
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 数据中存在实际问题,我会感到震惊,因为我相信现在有人会 运行 进入这个问题,因为我不可能是唯一一个尝试使用 @TypeAlias
与 spring-data-mongodb。更不用说这一切都适用于我使用的 spring 引导的先前版本(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 数据 mongodb 并且配置文件扩展了 AbstractMongoConfiguration:
@Configuration
@EnableMongoRepositories(basePackages = "com.mycompany")
@EnableMongoAuditing
public class MongoConfig extends AbstractMongoConfiguration
{
我重写 getMappingBasePackage()
方法来设置包扫描如下:
@Override
protected String getMappingBasePackage()
{
return "com.mycompany";
}
我一直在调试代码并注意到一些有趣的事情:
- 我在两个地方得到
java.lang.InstantiationError
。当我试图从 mongo 中读取引用了抽象 class (ParentClass) 的文档时,这两种情况都会发生。它试图实例化抽象 class 而不是找到我添加到 child classes. 的
@TypeAlias
注释
这是我的 ParentClass 的样子:
@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);
}
它加载包装器 class 很好,但是当它到达 child class 时,别名设置为 "ALIAS_TYPE_CHILD1",但以下值在类型缓存:
{
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。因为它不能实例化一个抽象 class 而爆炸了。这是堆栈跟踪:
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 数据中存在实际问题,我会感到震惊,因为我相信现在有人会 运行 进入这个问题,因为我不可能是唯一一个尝试使用 @TypeAlias
与 spring-data-mongodb。更不用说这一切都适用于我使用的 spring 引导的先前版本(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");
}