解决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);
这样一个映射就完成我们的需求了。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。如有错误或未考虑完全的地方,望不吝赐教。
相关文章
SpringBoot 整合 Shiro 密码登录与邮件验证码登录功能(多 Realm 认证)
这篇文章主要介绍了SpringBoot 整合 Shiro 密码登录与邮件验证码登录(多 Realm 认证),本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下2021-02-02
Redis Java 集成到 Spring Boot的详细过程
本文介绍了如何使用SpringBoot连接Redis,并展示了如何配置Redis服务地址、创建Controller类以及进行基本的Redis操作,如字符串、列表、集合、哈希和有序集合,感兴趣的朋友跟随小编一起看看吧2024-12-12


最新评论