優雅的使用BeanUtils對List集合的操作
- 2020 年 1 月 31 日
- 筆記
摘要
我們在Entity、Bo、Vo層數據間可能經常轉換數據,Entity對應的是持久層數據結構(一般是資料庫表的映射模型)、Bo對應的是業務層操作的數據結構、Vo就是Controller和客戶端交互的數據結構。在這些數據結構之間很大一部分屬性都可能會相同,我們在使用的時候會不斷的重新賦值。 如:客戶端傳輸管理員資訊的到Web層,我們會使用AdminVo接收,但是到了Service層時,我就需要使用AdminBo,這時候就需要把AdminVo實例的屬性一個一個賦值到AdminBo實例中。
BeanUtils
Spring 提供了 org.springframework.beans.BeanUtils 類進行快速賦值,如: AdminEntity類
public class AdminEntity{ private Integer id; private String password; private String username; private String userImg; .... //一些 Set Get方法 }
AdminVo類,因為是和客戶端打交道的,所以password屬性就不適合在這裡了
public class AdminVo{ private Integer id; private String username; private String userImg; .... //一些 Set Get方法 }
假如我們需要把AdminEntity實例屬性值賦值到AdminVo實例中(暫時忽略Bo層吧)
AdminEntity entity = ...; AdminVo vo = new AdminEntity(); // org.springframework.beans.BeanUtils BeanUtils.copyProperties(entity, vo); // 賦值
那麼這樣AdminVo實例中的屬性值就和AdminEntity實例中的屬性值一致了。 但是如果我們是一個集合的時候就不能這樣直接賦值了。如:
List<Admin> adminList = ...; List<AdminVo> adminVoList = new ArrayList<>(adminList.size()); BeanUtils.copyProperties(adminList, adminVoList); // 賦值失敗
這樣直接賦值是不可取的,由方法名(copyProperties)可知,只會複製他們的屬性值,那麼上述的adminList
屬性和adminVoList
的屬性是沒有半毛錢關係的。那麼怎麼解決了?
方式一(暴力解決,不推薦)
程式碼如下:
List<Admin> adminList = ...; List<AdminVo> adminVoList = new ArrayList<>(adminList.size()); for (Admin admin : adminList) { AdminVo adminVo = new AdminVo(); BeanUtils.copyProperties(admin, adminVo); adminVoList.add(adminVo); } return adminVoList;
雖然for
循環可以解決,但是一點都不優雅,這樣的程式碼也會陸續增多,程式碼多了,就會看出來了重複的地方,沒錯!就是for
循環賦值的地方,完全可以優化掉(程式碼寫多了一眼就看出來了)。那麼請看優雅的方式二
方式二 (優雅、推薦)
這也是我第一次寫泛型的程式碼,可能有待提高,如下: ColaBeanUtils類(Cola是我家的狗狗名,哈哈)
import org.springframework.beans.BeanUtils; public class ColaBeanUtils extends BeanUtils { public static <S, T> List<T> copyListProperties(List<S> sources, Supplier<T> target) { return copyListProperties(sources, target, null); } /** * @author Johnson * 使用場景:Entity、Bo、Vo層數據的複製,因為BeanUtils.copyProperties只能給目標對象的屬性賦值,卻不能在List集合下循環賦值,因此添加該方法 * 如:List<AdminEntity> 賦值到 List<AdminVo> ,List<AdminVo>中的 AdminVo 屬性都會被賦予到值 * S: 數據源類 ,T: 目標類::new(eg: AdminVo::new) */ public static <S, T> List<T> copyListProperties(List<S> sources, Supplier<T> target, ColaBeanUtilsCallBack<S, T> callBack) { List<T> list = new ArrayList<>(sources.size()); for (S source : sources) { T t = target.get(); copyProperties(source, t); list.add(t); if (callBack != null) { // 回調 callBack.callBack(source, t); } } return list; }
ColaBeanUtilsCallBack介面,使用java8的lambda表達式註解:
@FunctionalInterface public interface ColaBeanUtilsCallBack<S, T> { void callBack(S t, T s); }
使用方式如下:
List<AdminEntity> adminList = ...; return ColaBeanUtils.copyListProperties(adminList, AdminVo::new);
如果需要在循環中做處理(回調),那麼可使用lambda表達式:
List<Article> adminEntityList = articleMapper.getAllArticle(); List<ArticleVo> articleVoList = ColaBeanUtils.copyListProperties(adminEntityList , ArticleVo::new, (articleEntity, articleVo) -> { // 回調處理 }); return articleVoList;
簡直不要太簡單了!!!
總結
AdminVo::new
配合Supplier<T> target
這裡我完全是參考《Java核心技術第十版 卷一》的第八章泛型,因為之前看過有點印象,沒想到今天用到了。@FunctionalInterface
這個是java8的lambda
表達式的註解類,參考java.util.function.Consumer
介面。沒想到懵懵懂懂的,就把之前看過的知識寫出來了,驚呆了,哈哈。 程式碼如果雷同,純屬巧合。若泛型的程式碼還有可以改進的地方,可在下方留言,非常感謝,Thanks♪(・ω・)ノ。
個人部落格網址: https://colablog.cn/