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


摘要

我們在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/

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


免責聲明!

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



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