16

我正在为 MongoDB 使用 Spring-Data:

版本信息——org.mongodb.mongo-java-driver 2.10.1版本,org.springframework.data.spring-data-mongodb 1.2.1.RELEASE。

我有一个与此处定义的案例类似的案例,即(抱歉格式化...):

我刚开始使用 spring-data-mongodb 用 Ja​​va 开发一些应用程序,遇到了一些我无法解决的问题:

我有几个这样的文档 bean:

@Document(collection="myBeanBar")
public class BarImpl implements Bar {
    String id;
    Foo foo;
    // More fields and methods ... 
}

@Document
public class FooImpl implements Foo {
    String id;
    String someField;
    // some more fields and methods ...
} 

我有一个存储库类,它的方法简单地调用类似于这样的查找:

public List<? extends Bar> findByFooField(final String fieldValue) {
    Query query = Query.query(Criteria.where("foo.someField").is(fieldValue));
    return getMongoOperations().find(query, BarImpl.class);
} 

保存 Bar 效果很好,它会将它与 Foo 和 Bar 的“_class”属性一起保存在 mongo 中。但是,通过 Foo 中的某些属性查找会引发如下异常:

Exception in thread "main" java.lang.IllegalArgumentException: No
property someField found on test.Foo!
    at org.springframework.data.mapping.context.AbstractMappingContext.getPersistentPropertyPath(AbstractMappingContext.java:225)
    at org.springframework.data.mongodb.core.convert.QueryMapper.getPath(QueryMapper.java:202)
    at org.springframework.data.mongodb.core.convert.QueryMapper.getTargetProperty(QueryMapper.java:190)
    at org.springframework.data.mongodb.core.convert.QueryMapper.getMappedObject(QueryMapper.java:86)
    at org.springframework.data.mongodb.core.MongoTemplate.doFind(MongoTemplate.java:1336)
    at org.springframework.data.mongodb.core.MongoTemplate.doFind(MongoTemplate.java:1322)
    at org.springframework.data.mongodb.core.MongoTemplate.find(MongoTemplate.java:495)
    at org.springframework.data.mongodb.core.MongoTemplate.find(MongoTemplate.java:486)

给出的解决方案是在抽象类上使用 @TypeAlias 注释,它告诉框架使用特定的实现(在本例中为 FooImpl)。

就我而言,我有接口成员,而不是抽象成员:

@Document(collection="myBeanBar")
public class BarImpl implements Bar {
    String id;
    IFoo foo;
    // More fields and methods ...
}

我很不情愿在接口IFoo上放一个会给出默认实现的注解,相反我想告诉框架在实现BarImpl类的上下文中这个字段的默认实现是什么,类似于@JsonTypeInfo:

@Document(collection="myBeanBar") 
public class BarImpl implements Bar {
    String id;    

    @JsonTypeInfo(use = Id.CLASS, defaultImpl = FooImpl.class)
    IFoo foo; 

    // More fields and methods ... 
}

我找到了这个答案,它或多或少地说是为了避免使用接口。但我很高兴知道是否有更好的选择。

有任何想法吗?

谢谢!

4

3 回答 3

9

我的问题与问题类似,但抛出的异常有点不同:

Could not instantiate bean class [class name]: Specified class is an interface

当我的 DB 类的字段之一被声明为接口时,就会发生这种情况。保存此字段很好,但从 MongoDB 读取它时会引发异常。最后我找到了使用org.springframework.core.convert.converter.Converter.

两个步骤,1.构造一个实现的类Converter;2. 在 servlet 上下文中注册转换器。是的,您不必修改任何现有代码,例如添加注释。

下面是我的模型类,其中字段Data是一个接口:

@Document(collection="record")
public class Record {
    @Id
    private String id;

    // Data is an interface
    private Data data;

    // And some other fields and setter/getter methods of them
}

转换器:

@ReadingConverter
public class DataReadConverter implements Converter<DBObject, Data> {
    @Override
    public Data convert(DBObject source) {
        // Your implementation to parse the DBObject,
        // this object can be BasicDBObject or BasicDBList,
        // and return an object instance that implements Data.

        return null;
    }
}

最后要做的是注册转换器,我的配置是在xml中:

<mongo:mongo id="mongo" />

<mongo:db-factory mongo-ref="mongo" dbname="example" />

<mongo:mapping-converter>
    <mongo:custom-converters>
        <mongo:converter>
            <beans:bean class="com.example.DataReadConverter" />
        </mongo:converter>
    </mongo:custom-converters>
</mongo:mapping-converter>

<beans:bean id="mongoTemplate" class="org.springframework.data.mongodb.core.MongoTemplate">
    <beans:constructor-arg name="mongoDbFactory" ref="mongoDbFactory" />
    <beans:constructor-arg name="mongoConverter" ref="mappingConverter" />
</beans:bean>

部署应用程序并重试。它应该在接口字段上正确解析来自 MongoDB 的 DBObject。

我的 Spring MongoDB 应用程序的版本是:spring-*-4.1.0 和 spring-data-mongodb-1.6.0。

于 2014-11-24T08:32:11.763 回答
1

我收到了与@victor-wong 相同的错误信息

Could not instantiate bean class [class name]: Specified class is an interface

下面的代码用 Spring Boot 2.3.2and解决了这个问题spring-data-mongodb 3.0.2

转换器:

import org.bson.Document;
import org.springframework.core.convert.converter.Converter;
import org.springframework.data.convert.ReadingConverter;

@ReadingConverter
public class DataReadConverter implements Converter<Document, Data> {

    @Override
    public Data convert(Document source) {

        return new DataImpl(source.get("key"));
    }
}

最后要做的是注册转换器

    @Bean
    public MongoCustomConversions customConversions() {
        return new MongoCustomConversions(
                List.of(
                        new DataReadConverter()
                )
        );
    }

更多信息可以在这里找到:https ://jira.spring.io/browse/DATAMONGO-2391

于 2020-09-21T12:33:53.000 回答
0

将接口定义为数据对象中的字段确实是个坏主意。

接口意味着某些对象做某事的可能性,但不提供有关字段的任何信息。你真的需要使用接口吗?你能避免这种情况吗?即使使用抽象类定义也会是更好的主意。

PS当然,无论如何我的答案都不能被标记为正确答案。

于 2014-08-01T16:15:10.293 回答