tes*_*123 4 spring spring-data spring-data-mongodb
我目前正在使用 spring data mongodb 并且配置文件扩展了 AbstractMongoConfiguration:
@Configuration
@EnableMongoRepositories(basePackages = "com.mycompany")
@EnableMongoAuditing
public class MongoConfig extends AbstractMongoConfiguration
{
Run Code Online (Sandbox Code Playgroud)
我重写了getMappingBasePackage()将包设置为扫描的方法,如下所示:
@Override
protected String getMappingBasePackage()
{
return "com.mycompany";
}
Run Code Online (Sandbox Code Playgroud)
我一直在调试代码并注意到一些有趣的事情:
java.lang.InstantiationError. 当我尝试从 mongo 读取引用抽象类(ParentClass)的文档时,这两种情况都会发生。它试图实例化抽象类,而不是查找@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
{
...
Run Code Online (Sandbox Code Playgroud)
我的孩子的班级是这样的:
@Document
@JsonTypeName("JSON_TYPE_CHILD1")
@TypeAlias("ALIAS_TYPE_CHILD1")
public class Child1 extends ParentClass
{
...
Run Code Online (Sandbox Code Playgroud)
这就是我试图读入的 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"
}
Run Code Online (Sandbox Code Playgroud)
当我通过 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);
}
Run Code Online (Sandbox Code Playgroud)
它可以很好地加载包装类,但是当它到达子类时,别名将按其应有的方式设置为“ALIAS_TYPE_CHILD1”,但以下值位于 typeCache 中:
{
NONE=Optional.empty,
ALIAS_TYPE_CHILD1=Optional.empty,
com.mycompany.domain.job.Job=Optional[com.mycompany.domain.job.Job]
}
Run Code Online (Sandbox Code Playgroud)
由于键“ALIAS_TYPE_CHILD1”具有Optional.empty 作为值,因此代码无法获取要加载的正确目标类型,因此它使用作为ParentClass 的rawType。它会因为无法实例化抽象类而爆炸。这是堆栈跟踪:
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 ...
Run Code Online (Sandbox Code Playgroud)
奇怪的是,如果我插入一个包含@TypeAlias("ALIAS_TYPE_CHILD1") first的新文档,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]
}
Run Code Online (Sandbox Code Playgroud)
当我在插入后立即执行 findOne 时,我可以毫无错误地读取文档,因为它使用 Child1 来实例化 pojo 而不是 ParentClass。如果我尝试先读取,那么我是否在之后插入并不重要,因为typeCace在那里获取了错误的值,并且它会使用该值,直到您重新启动服务器为止。
我的猜测是配置或默认设置发生了变化。我能够解决所有其他升级问题,但这个问题让我感到困惑。如果 spring data 中存在实际问题,我会感到震惊,因为我确信现在有人会遇到这个问题,因为我不可能是唯一一个尝试使用@TypeAliasspring-data-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");
}
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
2809 次 |
| 最近记录: |