Ich werde meine eigene Frage beantworten. Wie in den Kommentaren erwähnt, besteht die Lösung darin, Converter
zu verwenden .
Hier ist ein Beispiel dafür, was ich mit meinem Klassenmodell erreichen wollte:
Ein Contact
kann entweder eine Person
sein oder eine Organisation
.
Wenn Sie spring-data-mongodb verwenden MongoRepository
um Daten gemäß Ihrem Entitätsmodell in Ihre Datenbank zu schreiben, eine _class
-Feld wird den Dokumentstämmen und komplexen Eigenschaftstypen hinzugefügt (siehe diesen Abschnitt
). Dieses Feld speichert den vollständig qualifizierten Namen der Java-Klasse und ermöglicht eine Begriffsklärung bei der Zuordnung vom MongoDb-Dokument zum Spring-Datenmodell.
Wenn Ihre App nur ein Dokument aus der Datenbank liest (kein _class
Felder), müssen Sie Spring Data mitteilen, welche Klasse beim Zuordnen eines Contact
instanziiert werden soll . Mit Spring-Data können Sie das standardmäßige Typzuordnungsverhalten mit Converter
anpassen . Verwendung von explizitem Converter
Standard überschreiben Mapping für die Klasse
. Sie müssen Ihre gesamte Klasse explizit zuordnen. Hier ist ein Beispiel für meinen ContactReadConverter :
@ReadingConverter
public class ContactReadConverter implements Converter<Document, Contact> {
@Override
public Contact convert(Document source) {
if (source.get("firstName") == null) {
Organisation organisation = new Organisation();
I18n name = new I18n();
name.setEn(source.get("name", Document.class).get("en", String.class));
name.setFr(source.get("name", Document.class).get("fr", String.class));
organisation.setName(name);
organisation.setAcronym(source.get("acronym", String.class));
organisation.setRole(source.get("role", String.class));
return organisation;
}
Person person = new Person();
person.setFirstName(source.get("firstName", String.class));
person.setLastName(source.get("lastName", String.class));
person.setRole(source.get("role", String.class));
person.setEmail(source.get("email", String.class));
person.setOrcId(source.get("orcId", String.class));
if (source.get("organisation") != null) {
Document sourceOrg = source.get("organisation", Document.class);
Organisation organisation = new Organisation();
organisation.setAcronym(sourceOrg.get("acronym", String.class));
organisation.setRole(sourceOrg.get("role", String.class));
if (sourceOrg.get("name") != null) {
I18n name = new I18n();
name.setFr(sourceOrg.get("name", Document.class).get("fr", String.class));
name.setEn(sourceOrg.get("name", Document.class).get("en", String.class));
organisation.setName(name);
}
person.setOrganisation(organisation);
}
return person;
}
}
Anschließend müssen neu definierte Konverter registriert werden:
@Configuration
public class DataportalApplicationConfig extends AbstractMongoConfiguration {
@Value("${spring.data.mongodb.uri}")
private String uri;
@Value("${spring.data.mongodb.database}")
private String database;
@Override
public MongoClient mongoClient() {
return new MongoClient(new MongoClientURI(uri));
}
@Override
protected String getDatabaseName() {
return database;
}
@Bean
@Override
public MongoCustomConversions customConversions() {
List<Converter<?, ?>> converterList = new ArrayList<>();
converterList.add(new ContactReadConverter());
return new MongoCustomConversions(converterList);
}
}
Hoffe es hilft.