解決springboot中@DynamicUpdate注解無效的問題
springboot 中 @DynamicUpdate 注解無效解決方案
遇到的問題
項目中使用 jpa,以前沒用過,所以踩坑在所難免。
在使用過程中,要更新一條記錄的某個字段,更新成功以后,發(fā)現(xiàn)整條記錄只剩下我更新的那個字段,其他的全部為空了。
瞬間明白,這種更新是全覆蓋,針對每個字段 update,實體類沒賦值的字段,也直接將空值 set 過去了。
尋求解決方案
正在慶幸這么容易就解決,突然發(fā)現(xiàn)并沒有這么簡單。
群眾的力量是無窮大的,我立刻就明白這個注解為什么無效,原來是搞錯了它的用途。
一孔解決方案
如剛才無數(shù)個箭頭指向的評論所說,用findOne查出原值,然后賦值想要修改的新的字段值。
思路很簡單,這里主要貼一下對象復制的代碼。將數(shù)據庫中查出的對象稱為target,包含要修改的字段的對象稱為source,當然,最后我們save的是修改之后的target
BeanCopyUtil:
import ch.qos.logback.core.joran.util.beans.BeanUtil; import org.springframework.beans.BeanUtils; import org.springframework.beans.BeanWrapper; import org.springframework.beans.BeanWrapperImpl; import java.util.HashSet; import java.util.Set; /** * created by xxx 2018/7/21 */ public class BeanCopyUtil { //source中的非空屬性復制到target中 public static <T> void beanCopy(T source, T target) { BeanUtils.copyProperties(source, target, getNullPropertyNames(source)); } //source中的非空屬性復制到target中,但是忽略指定的屬性,也就是說有些屬性是不可修改的(個人業(yè)務需要) public static <T> void beanCopyWithIngore(T source, T target, String... ignoreProperties) { String[] pns = getNullAndIgnorePropertyNames(source, ignoreProperties); BeanUtils.copyProperties(source, target, pns); } public static String[] getNullAndIgnorePropertyNames(Object source, String... ignoreProperties) { Set<String> emptyNames = getNullPropertyNameSet(source); for (String s : ignoreProperties) { emptyNames.add(s); } String[] result = new String[emptyNames.size()]; return emptyNames.toArray(result); } public static String[] getNullPropertyNames(Object source) { Set<String> emptyNames = getNullPropertyNameSet(source); String[] result = new String[emptyNames.size()]; return emptyNames.toArray(result); } public static Set<String> getNullPropertyNameSet(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()); } return emptyNames; } }
有了這個方法,在修改的時候就比較方便了,我的做法是在實體類中加一個方法:
//這里我設置【任務編號】和【創(chuàng)建人】不可修改 public void copy(Task task) { BeanCopyUtil.beanCopyWithIngore(task, this, "taskNum", "createPerson"); }
然后在service中update方法中調用:
@Transactional public Task updateTask(Task task) { try { if (task.getId() == null) { return null; } Task saveTask = taskRepository.findOne(task.getId()); saveTask.copy(task); return taskRepository.saveAndFlush(task); } catch (Exception e) { throw new CustomException(SERVER_ERROR, e); } }
總結
使用 springboot 時會遇到非常非常多的注解,這確實為開發(fā)省去了大量的時間,和很多沒有意義的體力勞動。但是在使用注解的時候,一定要弄清楚用途和用法 ,不然明明是用錯了,你還會覺得莫名其妙。
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
Java concurrency線程池之線程池原理(三)_動力節(jié)點Java學院整理
這篇文章主要為大家詳細介紹了Java concurrency線程池之線程池原理第三篇,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-06-06關于Spring自定義XML schema 擴展的問題(Spring面試高頻題)
今天給大家分享一道spring高頻率面試題關于Spring自定義XML schema 擴展的問題,今天以spring整合dubbo的實例給大家詳細講解下,感興趣的朋友跟隨小編一起看看吧2021-05-05