優雅的使用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/