解决springjpa的局部更新字段问题
问题描述:
使用springjpa更新数据时,有时候我们需要更新部分字段,对已有的内容保持不变,通常我们可以通过Spring提供的bean工具类BeanUtils来实现
解决方法:
BeanUtils复制对象,BeanUtils中的构造方法属性中可以通过传入更新时忽略的属性值来实现选择性复制原对象的字段。更新部分字段时,我们仅需要传入复制后的字段即可。
解析和实现:
BeanUtils的构造方法:
具体更新部分字段的步骤:
查询出待更新对象的原有信息
通过传入的更新的象去复制产生一个新对象,其中新对象中为null的字段不需要更新。
执行更新操作,操作对象时步骤2得出的复制对象。
代码如下:
public Result update(@RequestBody AppScene appScene, @PathVariable String id ){ AppScene target = appSceneService.findById(id); //数据库查出待更新对象 BeanUtils.copyProperties(appScene,target,getNullPropertyNames(appScene)); //使用更新对象的非空值去覆盖待更新对象 appSceneService.update(target ); //执行更新操作 return new Result(true,StatusCode.OK,"修改成功"); }
其中涉及的getNullPropertyNames方法作为工具类存在,具体如下:
public static String[] getNullPropertyNames (Object source) { final BeanWrapper src = new BeanWrapperImpl(source); java.beans.PropertyDescriptor[] pds = src.getPropertyDescriptors(); Set<String> emptyNames = new HashSet<String>(); for(java.beans.PropertyDescriptor pd : pds) { Object srcValue = src.getPropertyValue(pd.getName()); if (srcValue == null) emptyNames.add(pd.getName()); } String[] result = new String[emptyNames.size()]; return emptyNames.toArray(result); }
更新:
这是之前刚开始学习框架的时候遇到的问题,其实这个问题spring早就提供了对应的方法去解决,它提供了对应bean拷贝的方法BeanUtils.copyProperties,通过传入不同的值决定是否要忽略非空属性值的拷贝,现在已经没有必要自己手写了。
补充:Java Jpa选择性更新、部分字段更新工具类
使用Jpa自带的Save方法更新实体类时,会覆盖数据库中实体类原有内容。如果我们只想更新一部分字段或是选择性的更新,就只能另辟蹊径了。
这个工具类很好地弥补了这个不足,对于待更新实体类中有内容的字段会更新,为空的字段会采用原数据库中内容,下面是工具类代码(附使用方法)。
工具类代码:
import org.springframework.beans.BeanWrapper; import org.springframework.beans.BeanWrapperImpl; import java.util.HashSet; import java.util.Set; /** * jpa 部分字段更新方法 */ public class UpdateColumnUtil { public static String[] getNullPropertyNames(Object source) { final BeanWrapper src = new BeanWrapperImpl(source); java.beans.PropertyDescriptor[] pds = src.getPropertyDescriptors(); Set<String> emptyNames = new HashSet<>(); for (java.beans.PropertyDescriptor pd : pds) { Object srcValue = src.getPropertyValue(pd.getName()); if (srcValue == null) emptyNames.add(pd.getName()); } String[] result = new String[emptyNames.size()]; return emptyNames.toArray(result); } }
使用方法:
//首先从数据库查出待更新对象 Customer target = customerService.findById(customer.getCustId()); //使用更新对象的非空值去覆盖待更新对象 BeanUtils.copyProperties(customer, target, UpdateColumnUtil.getNullPropertyNames(customer)); //执行更新操作 save = customerService.save(target);
这样一个映射就完成我们的需求了。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。如有错误或未考虑完全的地方,望不吝赐教。
相关文章
BigDecimal的toString()、toPlainString()和toEngineeringString()区
使用BigDecimal进行打印的时候,经常会对BigDecimal提供的三个toString方法感到好奇,以下整理3个toString方法的区别及用法,需要的朋友可以参考下2023-08-08Junit 5中@ParameterizedTest与@EnumSource结合使用
今天小编就为大家分享一篇关于Junit 5中@ParameterizedTest与@EnumSource结合使用,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧2018-12-12
最新评论