12种 Vo2dto 方法,就 BeanUtil.copyProperties 压测数据最拉跨!

开发 前端
就像让你用一个属性拷贝工具,把vo转成dto,你用了哪呢,是 Apache 的还是 Spring 的,还是其他的什么,哪个效率最高?

[[427957]]

一、前言

为哈么,你的代码也就仅仅是能用而已?

没有技术深度、短缺知识储备、匮乏经验积累的前提下,怎么写代码?百度呀,遇到问题这搜一点,那查一块,不管它是什么原理还是适合哪种场景,先粘贴到自己的工程里,看,能跑了,能跑就行。那这样的代码也就仅仅是能用程度的交付,根本没有一定的质量保证,也更别提数据结构、算法逻辑和设计模式了,那看的编程资料刷的LeetCode,全歇菜了。

当你感觉看了很多资料又不会用的时候,会说什么,真卷,都学到这样了。但其实我并不觉对技术的深度挖掘、梳理全套的知识体系,一点点耕耘一点点收获是在卷。反而把看技术视频当成看电影一样轻松,不写案例就以为书看会了的爽,没有意义的缺少脑力思考机械式体力重复,才是卷,甚至很卷。

就像让你用一个属性拷贝工具,把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

1. get\set

  1. @Component 
  2. public class GetSetAssembler implements IAssembler<UserVO, UserDTO> { 
  3.  
  4.     @Override 
  5.     public UserDTO sourceToTarget(UserVO var) { 
  6.         UserDTO userDTO = new UserDTO(); 
  7.         userDTO.setUserId(var.getUserId()); 
  8.         userDTO.setUserNickName(var.getUserNickName()); 
  9.         userDTO.setCreateTime(var.getCreateTime()); 
  10.         return userDTO; 
  11.     } 
  12.  
  • 推荐:★★★☆☆
  • 性能:★★★★★
  • 手段:手写
  • 点评:其实这种方式也是日常使用的最多的,性能肯定是杠杠的,就是操作起来有点麻烦。尤其是一大堆属性的 VO 对象转换为 DTO 对象时候。但其实也有一些快捷的操作方式,比如你可以通过 Shift+Alt 选中所有属性,Shift+Tab 归并到一列,接下来在使用 Alt 选中这一列,批量操作粘贴 userDTO.set 以及快捷键大写属性首字母,最后切换到结尾补充括号和分号,最终格式化一下就搞定了。

2. json2Json

  1. @Component 
  2. public class Json2JsonAssembler implements IAssembler<UserVO, UserDTO> { 
  3.  
  4.     @Override 
  5.     public UserDTO sourceToTarget(UserVO var) { 
  6.         String strJson = JSON.toJSONString(var); 
  7.         return JSON.parseObject(strJson, UserDTO.class); 
  8.     } 
  9.  
  • 推荐:☆☆☆☆☆
  • 性能:★☆☆☆☆
  • 手段:把对象转JSON串,再把JSON转另外一个对象
  • 点评:这么写多半有点烧!

3. Apache copyProperties

  1. @Component 
  2. public class ApacheCopyPropertiesAssembler implements IAssembler<UserVO, UserDTO> { 
  3.  
  4.     @Override 
  5.     public UserDTO sourceToTarget(UserVO var) { 
  6.         UserDTO userDTO = new UserDTO(); 
  7.         try { 
  8.             BeanUtils.copyProperties(userDTO, var); 
  9.         } catch (IllegalAccessException | InvocationTargetException e) { 
  10.             e.printStackTrace(); 
  11.         } 
  12.         return userDTO; 
  13.     } 
  14.  
  • 推荐:☆☆☆☆☆
  • 性能:★☆☆☆☆
  • 手段:Introspector 机制获取到类的属性来进行赋值操作
  • 点评:有坑,兼容性交差,不建议使用

4. Spring copyProperties

  1. @Component 
  2. public class SpringCopyPropertiesAssembler implements IAssembler<UserVO, UserDTO> { 
  3.  
  4.     @Override 
  5.     public UserDTO sourceToTarget(UserVO var) { 
  6.         UserDTO userDTO = new UserDTO(); 
  7.         BeanUtils.copyProperties(var, userDTO); 
  8.         return userDTO; 
  9.     } 
  10.  
  • 推荐:★★★☆☆
  • 性能:★★★★☆
  • 手段:Introspector机制获取到类的属性来进行赋值操作
  • 点评:同样是反射的属性拷贝,Spring 提供的 copyProperties 要比 Apache 好用的多,只要你不用错,基本不会有啥问题。

5. Bean Mapping

  1. @Component 
  2. public class BeanMappingAssembler implements IAssembler<UserVO, UserDTO> { 
  3.  
  4.     @Override 
  5.     public UserDTO sourceToTarget(UserVO var) { 
  6.         UserDTO userDTO = new UserDTO(); 
  7.         BeanUtil.copyProperties(var, userDTO); 
  8.         return userDTO; 
  9.     } 
  10.  
  • 推荐:★★☆☆☆
  • 性能:★★★☆☆
  • 手段:属性拷贝
  • 点评:性能一般

6. Bean Mapping ASM

  1. @Component 
  2. public class BeanMappingAssembler implements IAssembler<UserVO, UserDTO> { 
  3.  
  4.     @Override 
  5.     public UserDTO sourceToTarget(UserVO var) { 
  6.         UserDTO userDTO = new UserDTO(); 
  7.         BeanUtil.copyProperties(var, userDTO); 
  8.         return userDTO; 
  9.     } 
  10.  
  • 推荐:★★★☆☆
  • 性能:★★★★☆
  • 手段:基于ASM字节码框架实现
  • 点评:与普通的 Bean Mapping 相比,性能有所提升,可以使用。

7. BeanCopier

  1. @Component 
  2. public class BeanCopierAssembler implements IAssembler<UserVO, UserDTO> { 
  3.  
  4.     @Override 
  5.     public UserDTO sourceToTarget(UserVO var) { 
  6.         UserDTO userDTO = new UserDTO(); 
  7.         BeanCopier beanCopier = BeanCopier.create(var.getClass(), userDTO.getClass(), false); 
  8.         beanCopier.copy(var, userDTO, null); 
  9.         return userDTO; 
  10.     } 
  11.  
  • 推荐:★★★☆☆
  • 性能:★★★★☆
  • 手段:基于CGlib字节码操作生成get、set方法
  • 点评:整体性能很不错,使用也不复杂,可以使用

8. Orika

  1. @Component 
  2. public class OrikaAssembler implements IAssembler<UserVO, UserDTO> { 
  3.  
  4.     /** 
  5.      * 构造一个MapperFactory 
  6.      */ 
  7.     private static MapperFactory mapperFactory = new DefaultMapperFactory.Builder().build(); 
  8.  
  9.     static { 
  10.         mapperFactory.classMap(UserDTO.class, UserVO.class) 
  11.                 .field("userId""userId")  // 字段不一致时可以指定 
  12.                 .byDefault() 
  13.                 .register(); 
  14.     } 
  15.  
  16.     @Override 
  17.     public UserDTO sourceToTarget(UserVO var) { 
  18.         return mapperFactory.getMapperFacade().map(var, UserDTO.class); 
  19.     } 
  20.  
  • 官网:https://orika-mapper.github.io/orika-docs/
  • 推荐:★★☆☆☆
  • 性能:★★★☆☆
  • 手段:基于字节码生成映射对象
  • 点评:测试性能不是太突出,如果使用的话需要把 MapperFactory 的构建优化成 Bean 对象

9. Dozer

  1. @Component 
  2. public class DozerAssembler implements IAssembler<UserVO, UserDTO> { 
  3.  
  4.     private static DozerBeanMapper mapper = new DozerBeanMapper(); 
  5.  
  6.     @Override 
  7.     public UserDTO sourceToTarget(UserVO var) { 
  8.         return mapper.map(var, UserDTO.class); 
  9.     } 
  10.  
  • 官网:http://dozer.sourceforge.net/documentation/gettingstarted.html
  • 推荐:★☆☆☆☆
  • 性能:★★☆☆☆
  • 手段:属性映射框架,递归的方式复制对象
  • 点评:性能有点差,不建议使用

10. ModelMapper

  1. @Component 
  2. public class ModelMapperAssembler implements IAssembler<UserVO, UserDTO> { 
  3.  
  4.     private static ModelMapper modelMapper = new ModelMapper(); 
  5.  
  6.     static { 
  7.         modelMapper.addMappings(new PropertyMap<UserVO, UserDTO>() { 
  8.             @Override 
  9.             protected void configure() { 
  10.                 // 属性值不一样可以自己操作 
  11.                 map().setUserId(source.getUserId()); 
  12.             } 
  13.         }); 
  14.     } 
  15.  
  16.     @Override 
  17.     public UserDTO sourceToTarget(UserVO var) { 
  18.         return modelMapper.map(var, UserDTO.class); 
  19.     } 
  20.  
  • 官网:http://modelmapper.org
  • 推荐:★★★☆☆
  • 性能:★★★☆☆
  • 手段:基于ASM字节码实现
  • 点评:转换对象数量较少时性能不错,如果同时大批量转换对象,性能有所下降

11. JMapper

  1. JMapper<UserDTO, UserVO> jMapper = new JMapper<>(UserDTO.class, UserVO.class, new JMapperAPI() 
  2.         .add(JMapperAPI.mappedClass(UserDTO.class) 
  3.                 .add(JMapperAPI.attribute("userId"
  4.                         .value("userId")) 
  5.                 .add(JMapperAPI.attribute("userNickName"
  6.                         .value("userNickName")) 
  7.                 .add(JMapperAPI.attribute("createTime"
  8.                         .value("createTime")) 
  9.         )); 
  • 官网:https://github.com/jmapper-framework/jmapper-core/wiki
  • 推荐:★★★★☆
  • 性能:★★★★★
  • 手段:Elegance, high performance and robustness all in one java bean mapper
  • 点评:速度真心可以,不过结合 SpringBoot 感觉有的一点点麻烦,可能姿势不对

12. MapStruct

  1. @Mapper(componentModel = "spring", unmappedTargetPolicy = ReportingPolicy.IGNORE, unmappedSourcePolicy = ReportingPolicy.IGNORE
  2. public interface UserDTOMapping extends IMapping<UserVO, UserDTO> { 
  3.  
  4.     /** 用于测试的单例 */ 
  5.     IMapping<UserVO, UserDTO> INSTANCE = Mappers.getMapper(UserDTOMapping.class); 
  6.  
  7.     @Mapping(target = "userId", source = "userId"
  8.     @Mapping(target = "createTime", dateFormat = "yyyy-MM-dd HH:mm:ss"
  9.     @Override 
  10.     UserDTO sourceToTarget(UserVO var1); 
  11.  
  12.     @Mapping(target = "userId", source = "userId"
  13.     @Mapping(target = "createTime", dateFormat = "yyyy-MM-dd HH:mm:ss"
  14.     @Override 
  15.     UserVO targetToSource(UserDTO var1); 
  16.  
  • 官网:https://github.com/mapstruct/mapstruct
  • 推荐:★★★★★
  • 性能:★★★★★
  • 手段:直接在编译期生成对应的get、set,像手写的代码一样
  • 点评:速度很快,不需要到运行期处理,结合到框架中使用方便

四、总结

其实对象属性转换的操作无非是基于反射、AOP、CGlib、ASM、Javassist 在编译时和运行期进行处理,再有好的思路就是在编译前生成出对应的get、set,就像手写出来的一样。

所以我更推荐我喜欢的 MapStruct,这货用起来还是比较舒服的,一种是来自于功能上的拓展性,易用性和兼容性。

 

无论哪种使用,都要做一下完整的测试和验证,不要上来就复制粘贴,否则你可能早早的就把挖好坑了,当然不一定是哪个兄弟来填坑了。

 

责任编辑:武晓燕 来源: bugstack虫洞栈
相关推荐

2022-12-09 07:53:20

vo2dto方法AOP

2021-12-15 10:26:13

Docker代码接口

2023-02-22 08:15:13

压测模拟计算

2022-09-13 08:40:51

DTOVOPO

2014-07-10 10:33:47

2021-03-08 08:54:12

驱动模型DTODO

2022-11-25 18:49:11

云原生

2019-08-19 00:14:12

网络测试带宽网络流量

2021-05-31 09:00:00

数据存储存储系统工具

2014-11-25 11:37:17

压测 软件测试

2014-03-06 09:52:23

数据备份恢复数据安全

2010-06-07 09:11:43

jQuery

2016-08-08 18:11:50

服务器压力测试

2019-11-19 07:56:30

MySQL压测数据表

2010-11-03 11:36:53

访问DB2表

2022-01-17 09:18:28

JMeter分布式压测

2023-01-16 08:09:22

PulsarMQ

2019-12-31 14:21:00

数据挖掘关系网络数据

2010-07-14 10:53:20

Web应用

2021-12-03 08:45:57

RocketMQ压测性能
点赞
收藏

51CTO技术栈公众号