摘要
我們在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);
if (callBack != null) {
// 回調
callBack.callBack(source, t);
}
list.add(t);
}
return list;
}
ColaBeanUtilsCallBack接口,使用java8的lambda表達式注解:
@FunctionalInterface
public interface ColaBeanUtilsCallBack<S, T> {
void callBack(S t, T s);
}
使用方式如下:
List<AdminEntity> adminList = ...;
List<ArticleVo> articleVoList = ColaBeanUtils.copyListProperties(adminList, AdminVo::new);
return articleVoList;
如果需要在循環中做處理(回調),那么可使用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/
如果我的文章幫助到您,可以關注我的微信公眾號,第一時間分享文章給您