優雅的使用BeanUtils對List集合的操作


摘自:https://www.cnblogs.com/Johnson-lin/p/12123012.html

優雅的使用BeanUtils對List集合的操作

 

摘要#

我們在Entity、Bo、Vo層數據間可能經常轉換數據,Entity對應的是持久層數據結構(一般是數據庫表的映射模型)、Bo對應的是業務層操作的數據結構、Vo就是Controller和客戶端交互的數據結構。在這些數據結構之間很大一部分屬性都可能會相同,我們在使用的時候會不斷的重新賦值。
如:客戶端傳輸管理員信息的到Web層,我們會使用AdminVo接收,但是到了Service層時,我就需要使用AdminBo,這時候就需要把AdminVo實例的屬性一個一個賦值到AdminBo實例中。

BeanUtils#

Spring 提供了 org.springframework.beans.BeanUtils 類進行快速賦值,如:
AdminEntity類

Copy
public class AdminEntity{ private Integer id; private String password; private String username; private String userImg; .... //一些 Set Get方法 }

AdminVo類,因為是和客戶端打交道的,所以password屬性就不適合在這里了

Copy
public class AdminVo{ private Integer id; private String username; private String userImg; .... //一些 Set Get方法 }

假如我們需要把AdminEntity實例屬性值賦值到AdminVo實例中(暫時忽略Bo層吧)

Copy
    AdminEntity entity = ...;
    AdminVo vo = new AdminEntity(); // org.springframework.beans.BeanUtils BeanUtils.copyProperties(entity, vo); // 賦值

那么這樣AdminVo實例中的屬性值就和AdminEntity實例中的屬性值一致了。
但是如果我們是一個集合的時候就不能這樣直接賦值了。如:

Copy
        List<Admin> adminList = ...;
        List<AdminVo> adminVoList = new ArrayList<>(adminList.size()); BeanUtils.copyProperties(adminList, adminVoList); // 賦值失敗

這樣直接賦值是不可取的,由方法名(copyProperties)可知,只會復制他們的屬性值,那么上述的adminList屬性和adminVoList的屬性是沒有半毛錢關系的。那么怎么解決了?

方式一(暴力解決,不推薦)#

代碼如下:

Copy
        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是我家的狗狗名,哈哈)

Copy
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表達式注解:

Copy
@FunctionalInterface public interface ColaBeanUtilsCallBack<S, T> { void callBack(S t, T s); }

使用方式如下:

Copy
    List<AdminEntity> adminList = ...;
        return ColaBeanUtils.copyListProperties(adminList, AdminVo::new);

如果需要在循環中做處理(回調),那么可使用lambda表達式:

Copy
        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/

如果我的文章幫助到您,可以關注我的微信公眾號,第一時間分享文章給您

微信公眾號

作者: Johnson木木

出處:https://www.cnblogs.com/Johnson-lin/p/12123012.html

本站使用「CC BY 4.0」創作共享協議,轉載請在文章明顯位置注明作者及出處。

 
 
 
0
0

 

 
 
 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM