解决springjpa的局部更新字段问题
问题描述:
使用springjpa更新数据时,有时候我们需要更新部分字段,对已有的内容保持不变,通常我们可以通过Spring提供的bean工具类BeanUtils来实现
解决方法:
BeanUtils复制对象,BeanUtils中的构造方法属性中可以通过传入更新时忽略的属性值来实现选择性复制原对象的字段。更新部分字段时,我们仅需要传入复制后的字段即可。
解析和实现:
BeanUtils的构造方法:
具体更新部分字段的步骤:
查询出待更新对象的原有信息
通过传入的更新的象去复制产生一个新对象,其中新对象中为null的字段不需要更新。
执行更新操作,操作对象时步骤2得出的复制对象。
代码如下:
publicResultupdate(@RequestBodyAppSceneappScene,@PathVariableStringid){ AppScenetarget=appSceneService.findById(id);//数据库查出待更新对象 BeanUtils.copyProperties(appScene,target,getNullPropertyNames(appScene));//使用更新对象的非空值去覆盖待更新对象 appSceneService.update(target);//执行更新操作 returnnewResult(true,StatusCode.OK,"修改成功"); }
其中涉及的getNullPropertyNames方法作为工具类存在,具体如下:
publicstaticString[]getNullPropertyNames(Objectsource){ finalBeanWrappersrc=newBeanWrapperImpl(source); java.beans.PropertyDescriptor[]pds=src.getPropertyDescriptors(); SetemptyNames=newHashSet (); for(java.beans.PropertyDescriptorpd:pds){ ObjectsrcValue=src.getPropertyValue(pd.getName()); if(srcValue==null)emptyNames.add(pd.getName()); } String[]result=newString[emptyNames.size()]; returnemptyNames.toArray(result); }
更新:
这是之前刚开始学习框架的时候遇到的问题,其实这个问题spring早就提供了对应的方法去解决,它提供了对应bean拷贝的方法BeanUtils.copyProperties,通过传入不同的值决定是否要忽略非空属性值的拷贝,现在已经没有必要自己手写了。
补充:JavaJpa选择性更新、部分字段更新工具类
使用Jpa自带的Save方法更新实体类时,会覆盖数据库中实体类原有内容。如果我们只想更新一部分字段或是选择性的更新,就只能另辟蹊径了。
这个工具类很好地弥补了这个不足,对于待更新实体类中有内容的字段会更新,为空的字段会采用原数据库中内容,下面是工具类代码(附使用方法)。
工具类代码:
importorg.springframework.beans.BeanWrapper; importorg.springframework.beans.BeanWrapperImpl; importjava.util.HashSet; importjava.util.Set; /** *jpa部分字段更新方法 */ publicclassUpdateColumnUtil{ publicstaticString[]getNullPropertyNames(Objectsource){ finalBeanWrappersrc=newBeanWrapperImpl(source); java.beans.PropertyDescriptor[]pds=src.getPropertyDescriptors(); SetemptyNames=newHashSet<>(); for(java.beans.PropertyDescriptorpd:pds){ ObjectsrcValue=src.getPropertyValue(pd.getName()); if(srcValue==null)emptyNames.add(pd.getName()); } String[]result=newString[emptyNames.size()]; returnemptyNames.toArray(result); } }
使用方法:
//首先从数据库查出待更新对象 Customertarget=customerService.findById(customer.getCustId()); //使用更新对象的非空值去覆盖待更新对象 BeanUtils.copyProperties(customer,target,UpdateColumnUtil.getNullPropertyNames(customer)); //执行更新操作 save=customerService.save(target);
这样一个映射就完成我们的需求了。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持毛票票。如有错误或未考虑完全的地方,望不吝赐教。