在使用Spring Data操作MongoDB中:
例如:我们有一个实体 Person
,有一个实体 EmailAddress
。
@Document(collection = "test_person") public class Person{ private String name; @DBRef private EmailAddress emailAddress; ... getter setter 方法 }
@Document(collection = "test_email") public class EmailAddress{ @Id private String id; private String value; ... getter setter 方法 }
当我们调用保存方法的时候:
public Person test(){ Person person = new Person(); person.setName("test"); EmailAddress emailAddress = new EmailAddress(); emailAddress.setId("5a05108d4dcc5dece03c9e69"); person.setEmailAddress(emailAddress); testRepository.save(person); return person; }
上述的代码中,返回的person只有id,没有emailAddress的其他值。
public Person test(){ Person person = new Person(); person.setName("test"); EmailAddress emailAddress = new EmailAddress(); emailAddress.setName("afafa"); person.setEmailAddress(emailAddress); testRepository.save(person); return person; }
上述的代码中,emailAddress不能被保存。
Spring Data MongoDB中存在一些生命周期事件,如:onBeforeConvert, onBeforeSave, onAfterSave, onAfterLoad and onAfterConvert等。我们可以继承 AbstractMappingEventListener
,然后重写这些方法,即可以实现。
/** * MongoDB级联控制 * Created by guanzhenxing on 2017/11/9. */ public class CascadeControlMongoEventListenerextends AbstractMongoEventListener<Object>{ @Autowired private MongoOperations mongoOperations; @Override public void onAfterSave(AfterSaveEvent<Object> event){ super.onAfterSave(event); Object source = event.getSource(); ReflectionUtils.doWithFields(source.getClass(), new CascadeAfterSaveCallback(source, mongoOperations)); } @Override public void onBeforeConvert(BeforeConvertEvent<Object> event){ super.onBeforeConvert(event); Object source = event.getSource(); ReflectionUtils.doWithFields(source.getClass(), new CascadeBeforeConvertCallback(source, mongoOperations)); } }
/** * 级联控制的回调 * Created by guanzhenxing on 2017/11/10. */ public class CascadeAfterSaveCallbackimplements ReflectionUtils.FieldCallback{ private Object source; private MongoOperations mongoOperations; public CascadeAfterSaveCallback(final Object source, final MongoOperations mongoOperations){ this.source = source; this.mongoOperations = mongoOperations; } @Override public void doWith(final Field field)throws IllegalArgumentException, IllegalAccessException { ReflectionUtils.makeAccessible(field); if (field.isAnnotationPresent(DBRef.class)) { final Object fieldValue = field.get(source); //获得值 if (fieldValue != null) { doCascadeLoad(field); } } } /** * 级联查询 * * @param field */ private void doCascadeLoad(Field field)throws IllegalAccessException { Object fieldValue = field.get(source); List<Field> idFields = ReflectionUtil.getAnnotationField(fieldValue, Id.class); //该方法是为了获得所有的被@Id注解的属性 if (idFields.size() == 1) { //只处理一个Id Object idValue = ReflectionUtil.getFieldValue(fieldValue, idFields.get(0).getName()); Object value = mongoOperations.findById(idValue, fieldValue.getClass()); //查询获得值 ReflectionUtil.setFieldValue(source, field.getName(), value); } } }
public class CascadeBeforeConvertCallbackimplements ReflectionUtils.FieldCallback{ private Object source; private MongoOperations mongoOperations; public CascadeBeforeConvertCallback(Object source, MongoOperations mongoOperations){ this.source = source; this.mongoOperations = mongoOperations; } @Override public void doWith(Field field)throws IllegalArgumentException, IllegalAccessException { ReflectionUtils.makeAccessible(field); if (field.isAnnotationPresent(DBRef.class)) { final Object fieldValue = field.get(source); //获得值 if (fieldValue != null) { doCascadeSave(field); } } } /** * 级联保存 * * @param field * @throws IllegalAccessException */ private void doCascadeSave(Field field)throws IllegalAccessException { if (field.isAnnotationPresent(CascadeSave.class)) { //如果有标识@CascadeSave注解 Object fieldValue = field.get(source); List<Field> idFields = ReflectionUtil.getAnnotationField(fieldValue, Id.class); if (idFields.size() == 1) { mongoOperations.save(fieldValue); } } } }
@Retention(RetentionPolicy.RUNTIME) @Target(ElementType.FIELD) public @interface CascadeSave { }
@Configuration public class MongoConfig{ @Bean public CascadeControlMongoEventListener userCascadingMongoEventListener(){ return new CascadeControlMongoEventListener(); } }
以上是核心代码。至此,我们就可以解决上述的问题了。
参考: http://www.baeldung.com/cascading-with-dbref-and-lifecycle-events-in-spring-data-mongodb