为了隔离变化,我们会将DAO查询出来的Entity(或者DO)和对前端提供的VO(或者DTO)隔离开来。多数情况的时候,它们的结构都是类似的;但是我们很不喜欢写很多冗长的b.setF1(a.getF1())这样的代码,于是我们需要简化对象拷贝方式。
对象拷贝——BeanUtils
Apache和Spring均有BeanUtils工具类, Apache的BeanUtils稳定性与效率都差些;Spring的BeanUtils比较稳定,不会因为量大了,耗时明显增加,故一般都使用Spring的BeanUtils。必须保证同名的两个成员变量类型相同。
public class BeanUtilsDemo {
public static void main(String[] args) {
UserDO userDO = DataUtil.createData();
log.info("拷贝前,userDO:{}", userDO);
UserDTO userDTO = new UserDTO();
BeanUtils.copyProperties(userDO,userDTO);
log.info("拷贝后,userDTO:{}", userDTO);
}
}
对象拷贝——BeanCopier
BeanCopier是用于在两个bean之间进行属性拷贝的。BeanCopier支持两种方式:
- 一种是不使用Converter的方式,仅对两个bean间属性名和类型完全相同的变量进行拷贝;
- 另一种则引入Converter,可以对某些特定属性值进行特殊操作。
1. 准备工作——依赖
该依赖非必须,因为Spring中已经集成了cglib,类全名是:org.springframework.cglib.beans.BeanCopier
<groupId>cglib</groupId>
<artifactId>cglib-nodep</artifactId>
<version>3.3.0</version>
</dependency>
2. 第一种情况:属性名称、类型都相同
直接可以进行复制即可,首先新建一个util的类。
package com.pay.common.util;
import org.apache.catalina.util.ParameterMap;
import org.springframework.cglib.beans.BeanCopier;
/**
* @ClassName: BeanCopier
* @Description: 做业务的时候,为了隔离变化,我们会将DAO查询出来的Entity和对前端提供的VO隔离开来。大概90%的时候,它们的结构都是类似的;但是我们很不喜欢写很多冗长的b.setF1(a.getF1())这样的代码,于是我们需要简化对象拷贝方式。
* @author: 郭秀志 jbcode@126.com
* @date: 2020/4/24 8:41
* @Copyright:
*/
public class BeanCopierUtil {
private static ParameterMap<Object, Object> BEAN_COPIERS = new ParameterMap<Object, Object>();
/**
* 带有缓存的beancopier进行属性的复制
*
* @param srcObj 源对象
* @param destObj 目标对象
*/
public static void copy(Object srcObj, Object destObj) {
String key = genKey(srcObj.getClass(), destObj.getClass());
BeanCopier copier = null;
if (!BEAN_COPIERS.containsKey(key)) {
copier = BeanCopier.create(srcObj.getClass(), destObj.getClass(), false);
BEAN_COPIERS.put(key, copier);
} else {
copier = (BeanCopier) BEAN_COPIERS.get(key);
}
copier.copy(srcObj, destObj, null);
}
/**
* 组装缓存beancopier的key
*
* @param srcClazz
* @param destClazz
* @return
*/
private static String genKey(Class<?> srcClazz, Class<?> destClazz) {
return srcClazz.getName() + destClazz.getName();
}
public static void main(String[] args) {
//使用示例
/*BzSetupWorktime worktime = new BzSetupWorktime();
worktime.setCopierValue("guoxiuzhi");//这个属性在目标bean没有所以不会被复制进去。
worktime.setId((long) 8888);
worktime.setWtimeId("guoxiuzhi");
BzSetupWorktimeEntity bzSetupWorktimeEntity = new BzSetupWorktimeEntity();
BeanCopierUtil.copy(worktime, bzSetupWorktimeEntity);
return bzSetupWorktimeEntity;*/
}
}
在具体的业务中使用util类,此段示例的代码见上面main方法注释部分.
3. 第二种情况:属性名称相同、类型不同
BeanCopier只拷贝名称和类型都相同的属性。即使源类型是原始类型(int, short和char等),目标类型是其包装类型(Integer, Short和Character等),或反之:都不会被拷贝。
此时我们可以通过实现Converter接口来自定义转换器。
总结
BeanUtils与BeanCopier速度对比
序号 | 场景 | 耗时(10000次调用) | 原理 |
---|---|---|---|
1 | BeanUtils | 232ms | 反射 |
2 | BeanCopier | 116ms | 修改字节码 |
3 | BeanCopier(加缓存) | 6ms | 修改字节码 |