解决springboot中@DynamicUpdate注解无效的问题

时间:2022-09-20 23:51:42

springboot 中 @DynamicUpdate 注解无效解决方案

遇到的问题

项目中使用 jpa,以前没用过,所以踩坑在所难免。

在使用过程中,要更新一条记录的某个字段,更新成功以后,发现整条记录只剩下我更新的那个字段,其他的全部为空了。

瞬间明白,这种更新是全覆盖,针对每个字段 update,实体类没赋值的字段,也直接将空值 set 过去了。

寻求解决方案

解决springboot中@DynamicUpdate注解无效的问题

正在庆幸这么容易就解决,突然发现并没有这么简单。

解决springboot中@DynamicUpdate注解无效的问题

群众的力量是无穷大的,我立刻就明白这个注解为什么无效,原来是搞错了它的用途。

一孔解决方案

如刚才无数个箭头指向的评论所说,用findOne查出原值,然后赋值想要修改的新的字段值。

思路很简单,这里主要贴一下对象复制的代码。将数据库中查出的对象称为target,包含要修改的字段的对象称为source,当然,最后我们save的是修改之后的target

BeanCopyUtil:

  1. import ch.qos.logback.core.joran.util.beans.BeanUtil;
  2. import org.springframework.beans.BeanUtils;
  3. import org.springframework.beans.BeanWrapper;
  4. import org.springframework.beans.BeanWrapperImpl;
  5. import java.util.HashSet;
  6. import java.util.Set;
  7. /**
  8. * created by xxx 2018/7/21
  9. */
  10. public class BeanCopyUtil {
  11. //source中的非空属性复制到target中
  12. public static <T> void beanCopy(T source, T target) {
  13. BeanUtils.copyProperties(source, target, getNullPropertyNames(source));
  14. }
  15. //source中的非空属性复制到target中,但是忽略指定的属性,也就是说有些属性是不可修改的(个人业务需要)
  16. public static <T> void beanCopyWithIngore(T source, T target, String... ignoreProperties) {
  17. String[] pns = getNullAndIgnorePropertyNames(source, ignoreProperties);
  18. BeanUtils.copyProperties(source, target, pns);
  19. }
  20. public static String[] getNullAndIgnorePropertyNames(Object source, String... ignoreProperties) {
  21. Set<String> emptyNames = getNullPropertyNameSet(source);
  22. for (String s : ignoreProperties) {
  23. emptyNames.add(s);
  24. }
  25. String[] result = new String[emptyNames.size()];
  26. return emptyNames.toArray(result);
  27. }
  28. public static String[] getNullPropertyNames(Object source) {
  29. Set<String> emptyNames = getNullPropertyNameSet(source);
  30. String[] result = new String[emptyNames.size()];
  31. return emptyNames.toArray(result);
  32. }
  33. public static Set<String> getNullPropertyNameSet(Object source) {
  34. final BeanWrapper src = new BeanWrapperImpl(source);
  35. java.beans.PropertyDescriptor[] pds = src.getPropertyDescriptors();
  36. Set<String> emptyNames = new HashSet<>();
  37. for (java.beans.PropertyDescriptor pd : pds) {
  38. Object srcValue = src.getPropertyValue(pd.getName());
  39. if (srcValue == null) emptyNames.add(pd.getName());
  40. }
  41. return emptyNames;
  42. }
  43. }

有了这个方法,在修改的时候就比较方便了,我的做法是在实体类中加一个方法:

  1. //这里我设置【任务编号】和【创建人】不可修改
  2. public void copy(Task task) {
  3. BeanCopyUtil.beanCopyWithIngore(task, this, "taskNum", "createPerson");
  4. }

然后在service中update方法中调用:

  1. @Transactional
  2. public Task updateTask(Task task) {
  3. try {
  4. if (task.getId() == null) {
  5. return null;
  6. }
  7. Task saveTask = taskRepository.findOne(task.getId());
  8. saveTask.copy(task);
  9. return taskRepository.saveAndFlush(task);
  10. } catch (Exception e) {
  11. throw new CustomException(SERVER_ERROR, e);
  12. }
  13. }

总结

使用 springboot 时会遇到非常非常多的注解,这确实为开发省去了大量的时间,和很多没有意义的体力劳动。但是在使用注解的时候,一定要弄清楚用途和用法 ,不然明明是用错了,你还会觉得莫名其妙。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

原文链接:https://blog.csdn.net/m0_37659871/article/details/81143199