一、前言
为哈么,你的代码也就仅仅是能用而已?
让你用一个属性拷贝工具,把vo转成dto
,你用了哪呢,是 Apache 的还是 Spring 的,还是其他的什么,哪个效率最高?接下来我们来用数据验证下,并提供出各种案例的使用对比
二、性能测试对比
在 Java 系统工程开发过程中,都会有各个层之间的对象转换,比如 VO、DTO、PO、VO 等,而如果都是手动get、set
又太浪费时间,还可能操作错误,所以选择一个自动化工具会更加方便。
目前我整理出,用于对象属性转换有12种,包括:普通的getset、json2Json、Apache属性拷贝、Spring属性拷贝、bean-mapping、bean-mapping-asm、BeanCopier、Orika、Dozer、ModelMapper、JMapper、MapStruct 接下来我们分别测试这11种属性转换操作分别在一百次
、一千次
、一万次
、十万次
、一百万次
时候的性能时间对比。
BeanUtils.copyProperties
是大家代码里最常出现的工具类,但只要你不把它用错成 Apache
包下的,而是使用 Spring 提供的,就基本还不会对性能造成多大影响。
但如果说性能更好,可替代手动get、set
的,还是 MapStruct
更好用,因为它本身就是在编译期生成get、set
代码,和我们写get、set
一样。
其他一些组件包主要基于 AOP
、ASM
、CGlib
,的技术手段实现的,所以也会有相应的性能损耗。
三、12种转换案例
源码:https://github.com/fuzhengwei/guide-vo2dto
描述:在案例工程下创建 interfaces.assembler 包,定义 IAssembler<SOURCE, TARGET>#sourceToTarget(SOURCE var)
接口,提供不同方式的对象转换操作类实现,学习的过程中可以直接下载运行调试。
1. get\set
1 2 3 4 5 6 7 8 9 10 11 12
| @Component public class GetSetAssembler implements IAssembler<UserVO, UserDTO\> {
@Override public UserDTO sourceToTarget(UserVO var) { UserDTO userDTO = new UserDTO(); userDTO.setUserId(var.getUserId()); userDTO.setUserNickName(var.getUserNickName()); userDTO.setCreateTime(var.getCreateTime()); return userDTO; } }
|
2. json2Json
1 2 3 4 5 6 7 8 9
| @Component public class Json2JsonAssembler implements IAssembler<UserVO, UserDTO\> {
@Override public UserDTO sourceToTarget(UserVO var) { String strJson = JSON.toJSONString(var); return JSON.parseObject(strJson, UserDTO.class); } }
|
3. Apache copyProperties
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| @Component public class ApacheCopyPropertiesAssembler implements IAssembler<UserVO, UserDTO\> {
@Override public UserDTO sourceToTarget(UserVO var) { UserDTO userDTO = new UserDTO(); try { BeanUtils.copyProperties(userDTO, var); } catch (IllegalAccessException | InvocationTargetException e) { e.printStackTrace(); } return userDTO; } }
|
4. Spring copyProperties
1 2 3 4 5 6 7 8 9 10
| @Component public class SpringCopyPropertiesAssembler implements IAssembler<UserVO, UserDTO\> {
@Override public UserDTO sourceToTarget(UserVO var) { UserDTO userDTO = new UserDTO(); BeanUtils.copyProperties(var, userDTO); return userDTO; } }
|
5. Bean Mapping
1 2 3 4 5 6 7 8 9 10
| @Component public class BeanMappingAssembler implements IAssembler<UserVO, UserDTO\> {
@Override public UserDTO sourceToTarget(UserVO var) { UserDTO userDTO = new UserDTO(); BeanUtil.copyProperties(var, userDTO); return userDTO; } }
|
推荐:★★☆☆☆
性能:★★★☆☆
手段:属性拷贝
点评:性能一般
6. Bean Mapping ASM
1 2 3 4 5 6 7 8 9 10
| @Component public class BeanMappingAssembler implements IAssembler<UserVO, UserDTO\> {
@Override public UserDTO sourceToTarget(UserVO var) { UserDTO userDTO = new UserDTO(); BeanUtil.copyProperties(var, userDTO); return userDTO; } }
|
7. BeanCopier
1 2 3 4 5 6 7 8 9 10 11
| @Component public class BeanCopierAssembler implements IAssembler<UserVO, UserDTO\> {
@Override public UserDTO sourceToTarget(UserVO var) { UserDTO userDTO = new UserDTO(); BeanCopier beanCopier = BeanCopier.create(var.getClass(), userDTO.getClass(), false); beanCopier.copy(var, userDTO, null); return userDTO; } }
|
8. Orika
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| @Component public class OrikaAssembler implements IAssembler<UserVO, UserDTO\> {
private static MapperFactory mapperFactory = new DefaultMapperFactory.Builder().build();
static { mapperFactory.classMap(UserDTO.class, UserVO.class) .field("userId", "userId") .byDefault() .register(); }
@Override public UserDTO sourceToTarget(UserVO var) { return mapperFactory.getMapperFacade().map(var, UserDTO.class); } }
|
9. Dozer
1 2 3 4 5 6 7 8 9 10
| @Component public class DozerAssembler implements IAssembler<UserVO, UserDTO\> {
private static DozerBeanMapper mapper = new DozerBeanMapper();
@Override public UserDTO sourceToTarget(UserVO var) { return mapper.map(var, UserDTO.class); } }
|
10. ModelMapper
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| @Component public class ModelMapperAssembler implements IAssembler<UserVO, UserDTO\> {
private static ModelMapper modelMapper = new ModelMapper();
static { modelMapper.addMappings(new PropertyMap<UserVO, UserDTO>() { @Override protected void configure() {
map().setUserId(source.getUserId()); } }); }
@Override public UserDTO sourceToTarget(UserVO var) { return modelMapper.map(var, UserDTO.class); } }
|
11. JMapper
1 2 3 4 5 6 7 8 9
| JMapper<UserDTO, UserVO> jMapper = new JMapper<>(UserDTO.class, UserVO.class, new JMapperAPI() .add(JMapperAPI.mappedClass(UserDTO.class) .add(JMapperAPI.attribute("userId") .value("userId")) .add(JMapperAPI.attribute("userNickName") .value("userNickName")) .add(JMapperAPI.attribute("createTime") .value("createTime")) ));
|
12. MapStruct
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| @Mapper(componentModel = "spring", unmappedTargetPolicy = ReportingPolicy.IGNORE, unmappedSourcePolicy = ReportingPolicy.IGNORE) public interface UserDTOMapping extends IMapping<UserVO, UserDTO\> {
IMapping<UserVO, UserDTO> INSTANCE = Mappers.getMapper(UserDTOMapping.class);
@Mapping(target = "userId", source = "userId") @Mapping(target = "createTime", dateFormat = "yyyy-MM-dd HH:mm:ss") @Override UserDTO sourceToTarget(UserVO var1);
@Mapping(target = "userId", source = "userId") @Mapping(target = "createTime", dateFormat = "yyyy-MM-dd HH:mm:ss") @Override UserVO targetToSource(UserDTO var1);
}
|
四、总结
其实对象属性转换的操作无非是基于反射、AOP、CGlib、ASM、Javassist 在编译时和运行期进行处理,再有好的思路就是在编译前生成出对应的get、set,就像手写出来的一样。
所以我更推荐我喜欢的 MapStruct,这货用起来还是比较舒服的,一种是来自于功能上的拓展性,易用性和兼容性。
无论哪种使用,都要做一下完整的测试和验证,不要上来就复制粘贴,否则你可能早早的就把挖好坑了,当然不一定是哪个兄弟来填坑了。