开发必备:主流对象复制框架使用与比较

在 Java 生态中,有很多第三方框架可以进行对象复制,分别是Apache的BeanUtils和PropertyUtils、Spring的BeanUtils、Cglib的BeanCopier等,下面我们一起来了解一下吧!

一、摘要

对象复制,在实际开发中也是经常需要使用到的,想必最傻瓜式的就是创建一个对象,然后一个一个属性进行set/get,但是如果面对一个对象有100多个属性的情况,显然这种方法效率底、而且容易出错!

当然也有办法,使用 Java 提供的克隆方法,也可以实现对象拷贝,但是 Java 提供的对象拷贝,只能进行浅克隆,如果要进行深度克隆,可以使用序列化流实现对象深度复制。

虽然上述方法,可以解决对象的复制,假如我们有两个类,属性都基本一样,现在的你想要实现属性参数值复制,应该怎么解决呢?

其实解决思路很简单,就是找到两个类的名称、类型是否一致,如果一致,就将要复制的对象内容拷贝到另一个对象的属性上。

当然,在 Java 生态中已经有很多成熟的第三方框架实现了这一功能,在这里,介绍一下 java 的 Bean 复制框架,分别有ApacheBeanUtilsPropertyUtilsSpringBeanUtilsCglibBeanCopier等。

也不多说了,直接撸代码!

二、方法介绍

下面我们来创建两个类FromBeanToBean,两个类除了名称不一样,属性都一样,如下:

public class FromBean {

    private String id;

    private String name;

    private Integer age;

    private Date createTime;

    private BigDecimal money;

    //... 省略 setter 和 getter
}
public class ToBean {

    private String id;

    private String name;

    private Integer age;

    private Date createTime;

    private BigDecimal money;

    //... 省略 setter 和 getter
}

为了后面更好的比较各个框架的性能,这里使用策略模式,来实现各个方法,先新建一个对象复制接口,如下:

public interface BeanCopyService {

    /**
     * 使用的工具包名称
     * @return
     */
    String methodName();

    /**
     * 执行复制
     * @param fromBean
     * @return
     */
    ToBean copyProperty(FromBean fromBean) throws Exception;
}

新建一个对象复制策略类,如下:

public class BeanCopyStragegy {

    /**
     * 进行复制
     * @param beanCopyService
     * @param fromBean
     */
    public void execute(BeanCopyService beanCopyService, FromBean fromBean) throws Exception {
        System.out.println("=========================");
        System.out.println("使用的工具包:" + beanCopyService.methodName());
        ToBean toBean = beanCopyService.copyProperty(fromBean);
        System.out.println("复制后的对象:" + JSON.toJSONString(toBean));
    }

}

最后,创建一个测试类,如下:

public class BeanCopyClient {

    public static void main(String[] args) throws Exception {
        //初始化一个源对象bean
        FromBean fromBean = new FromBean();
        fromBean.setId("ID0001");
        fromBean.setName("张三");
        fromBean.setAge(18);
        fromBean.setCreateTime(new Date());
        fromBean.setMoney(new BigDecimal(100));

        //apache的BeanUtils
        BeanCopyService apacheBeanUtils = new BeanCopyService() {
            @Override
            public String methodName() {
                return "apache BeanUtils";
            }

            @Override
            public ToBean copyProperty(FromBean fromBean) throws Exception {
                ToBean toBean = new ToBean();
                org.apache.commons.beanutils.BeanUtils.copyProperties(toBean,fromBean);
                return toBean;
            }
        };

        //apache的PropertyUtils
        BeanCopyService apachePropertyUtils = new BeanCopyService() {
            @Override
            public String methodName() {
                return "apache PropertyUtils";
            }

            @Override
            public ToBean copyProperty(FromBean fromBean) throws Exception {
                ToBean toBean = new ToBean();
                org.apache.commons.beanutils.PropertyUtils.copyProperties(toBean,fromBean);
                return toBean;
            }
        };

        //spring的BeanUtils
        BeanCopyService springBeanUtils = new BeanCopyService() {
            @Override
            public String methodName() {
                return "spring BeanUtils";
            }

            @Override
            public ToBean copyProperty(FromBean fromBean) throws Exception {
                ToBean toBean = new ToBean();
                org.springframework.beans.BeanUtils.copyProperties(fromBean, toBean);
                return toBean;
            }
        };

        //cglib的BeanCopier
        BeanCopyService cglibBeanCopier = new BeanCopyService() {
            @Override
            public String methodName() {
                return "cglib BeanCopier";
            }

            @Override
            public ToBean copyProperty(FromBean fromBean) throws Exception {
                ToBean toBean = new ToBean();
                net.sf.cglib.beans.BeanCopier.create(FromBean.class, ToBean.class, false).copy(fromBean, toBean, null);
                return toBean;
            }
        };
        System.out.println("源对象的内容:" + JSON.toJSONString(fromBean));
       //进行测试
       BeanCopyStragegy stragegy = new BeanCopyStragegy();
       stragegy.execute(apacheBeanUtils, fromBean);//apache的BeanUtils
       stragegy.execute(apachePropertyUtils, fromBean);//apache的PropertyUtils
       stragegy.execute(springBeanUtils, fromBean);//spring的BeanUtils
       stragegy.execute(cglibBeanCopier, fromBean);//cglib的BeanCopier
    }
}

运行程序,输出结果如下:

其中 apache 提供了2个类来实现对象复制,分别是BeanUtilsPropertyUtils,这两者的区别在于BeanUtils提供类型转换功能,即发现两个JavaBean的同名属性为不同类型时,在支持的数据类型范围内进行转换,而PropertyUtils不支持这个功能!

关于这点,我们将ToBean类中的money类型修改成String,如下:

public class ToBean {

    private String id;

    private String name;

    private Integer age;

    private Date createTime;

    private String money;//修改money的类型为String

    //... 省略 setter 和 getter
}

在运行程序,输入结果如下:

从结果可以看出,apacheBeanUtils依然可以完整复制,但是PropertyUtils报错!

springBeanUtilscglibBeanCopier,因为属性对应的类型不一致,直接跳过复制!

既然,四者都可以复制,我们从性能角度来观察一下各个框架复制的效率如下,我们将BeanCopyStragegy类进行改造,如下:

public class BeanCopyStragegy {

    private Integer count;

    public BeanCopyStragegy(Integer count) {
        this.count = count;
    }

    /**
     * 进行复制
     * @param beanCopyService
     * @param fromBean
     */
    public void execute(BeanCopyService beanCopyService, FromBean fromBean) throws Exception {
        System.out.print("使用的工具包:" + beanCopyService.methodName());
        long start = System.currentTimeMillis();
        for (int i = 0; i < count; i++) {
            ToBean toBean = beanCopyService.copyProperty(fromBean);
        }
        System.out.print("循环复制对象"+count+"次,耗时:" + (System.currentTimeMillis() - start) +"\n");
    }
}

在测试的时候,传入循环次数!

//传入循环次数,进行测试
BeanCopyStragegy stragegy = new BeanCopyStragegy(10);

我们分别传入10100100010000100000,最终各个框架耗时结果如下:

可能每台机器测试结果有差异,从结果可以得出如下结论:

  • 在低频次下,apache的PropertyUtils效率最高,在高频次下,cglib BeanCopier效率最高;

  • apache的BeanUtils,在四个框架中效率最低;

  • 在高频次下,spring的BeanUtils的效率仅次于cglib BeanCopier;

三、总结

因为apacheBeanUtils在进行复制的时候,在支持的数据类型范围内进行转换,所有性能会有些损失,对于数据结构严谨的复制,不建议采用!

< END >

如果大家喜欢我们的文章,欢迎大家转发,点击在看让更多的人看到。也欢迎大家热爱技术和学习的朋友加入的我们的知识星球当中,我们共同成长,进步。

延伸阅读

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值