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

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)

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

  1. 我有两个地方可以得到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)配合得很好。

欢迎对下一步尝试的任何想法或建议。我只是不知道下一步该做什么。

tes*_*123 7

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

重写此方法可以解决该问题:

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