解決springjpa的局部更新字段問題
問題描述:
使用springjpa更新數(shù)據(jù)時,有時候我們需要更新部分字段,對已有的內(nèi)容保持不變,通常我們可以通過Spring提供的bean工具類BeanUtils來實現(xiàn)
解決方法:
BeanUtils復(fù)制對象,BeanUtils中的構(gòu)造方法屬性中可以通過傳入更新時忽略的屬性值來實現(xiàn)選擇性復(fù)制原對象的字段。更新部分字段時,我們僅需要傳入復(fù)制后的字段即可。
解析和實現(xiàn):
BeanUtils的構(gòu)造方法:
具體更新部分字段的步驟:
查詢出待更新對象的原有信息
通過傳入的更新的象去復(fù)制產(chǎn)生一個新對象,其中新對象中為null的字段不需要更新。
執(zhí)行更新操作,操作對象時步驟2得出的復(fù)制對象。
代碼如下:
public Result update(@RequestBody AppScene appScene, @PathVariable String id ){
AppScene target = appSceneService.findById(id); //數(shù)據(jù)庫查出待更新對象
BeanUtils.copyProperties(appScene,target,getNullPropertyNames(appScene)); //使用更新對象的非空值去覆蓋待更新對象
appSceneService.update(target ); //執(zhí)行更新操作
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);
}
更新:
這是之前剛開始學(xué)習(xí)框架的時候遇到的問題,其實這個問題spring早就提供了對應(yīng)的方法去解決,它提供了對應(yīng)bean拷貝的方法BeanUtils.copyProperties,通過傳入不同的值決定是否要忽略非空屬性值的拷貝,現(xiàn)在已經(jīng)沒有必要自己手寫了。
補(bǔ)充:Java Jpa選擇性更新、部分字段更新工具類
使用Jpa自帶的Save方法更新實體類時,會覆蓋數(shù)據(jù)庫中實體類原有內(nèi)容。如果我們只想更新一部分字段或是選擇性的更新,就只能另辟蹊徑了。
這個工具類很好地彌補(bǔ)了這個不足,對于待更新實體類中有內(nèi)容的字段會更新,為空的字段會采用原數(shù)據(jù)庫中內(nèi)容,下面是工具類代碼(附使用方法)。
工具類代碼:
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);
}
}
使用方法:
//首先從數(shù)據(jù)庫查出待更新對象 Customer target = customerService.findById(customer.getCustId()); //使用更新對象的非空值去覆蓋待更新對象 BeanUtils.copyProperties(customer, target, UpdateColumnUtil.getNullPropertyNames(customer)); //執(zhí)行更新操作 save = customerService.save(target);
這樣一個映射就完成我們的需求了。
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。
相關(guān)文章
Java 數(shù)組復(fù)制clone方法實現(xiàn)詳解
這篇文章主要介紹了Java 數(shù)組復(fù)制clone方法實現(xiàn)詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2019-11-11
SpringBoot 整合 Shiro 密碼登錄與郵件驗證碼登錄功能(多 Realm 認(rèn)證)
這篇文章主要介紹了SpringBoot 整合 Shiro 密碼登錄與郵件驗證碼登錄(多 Realm 認(rèn)證),本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-02-02
Redis Java 集成到 Spring Boot的詳細(xì)過程
本文介紹了如何使用SpringBoot連接Redis,并展示了如何配置Redis服務(wù)地址、創(chuàng)建Controller類以及進(jìn)行基本的Redis操作,如字符串、列表、集合、哈希和有序集合,感興趣的朋友跟隨小編一起看看吧2024-12-12
java Executors工具類的相關(guān)方法使用創(chuàng)建
這篇文章主要為大家介紹了java Executors工具類的相關(guān)方法使用創(chuàng)建,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11

