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

如果我的文章帮助到您,可以关注我的微信公众号,第一时间分享文章给您
微信公众号

posted @ 2019-12-31 10:13  Johnson木木  阅读(11642)  评论(3编辑  收藏  举报