面试官:对象映射的方式对比与选择?
yuyutoo 2024-10-29 17:28 3 浏览 0 评论
项目开发过程中,经常需要编写model之间的转换,最常见的有:
- 实体转DTO
- DTO转实体
- VO转...
举个例子:
// 实体:User
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class User {
private Integer id;
private String email;
private String username;
private String password;
private Integer gender;
private Date birthday;
}
// DTO:UserRegisterReq
@Data
public class UserRegisterReq {
private String username;
private String password;
private String confirmPassword;
private String email;
}
其中:
- UserRegisterReq是用户注册时,Controller层的请求入参
- User是用户实体
在执行注册时,我们需要将UserRegisterReq转换成User对象,再存储到数据库。此时,我们往往会编写类似如下的代码:
@PostMapping("/users/reg")
public void reg(@RequestBody UserRegisterReq userRegisterReq) {
// 省略password 与 confirmPassword等值判断
User user = new User();
user.setEmail(userRegisterReq.getEmail());
user.setPassword(userRegisterReq.getPassword());
user.setUsername(userRegisterReq.getUsername());
// 保存user...
}
如上的代码虽然可行,但是如果类里面的field非常多,那么就会比较麻烦——我们写了一堆代码,只不过是为了实现对象的转换而已。
方法一、IDEA插件快速转换
IDEA提供GenerateAllSetter插件,可帮助我们快速生成上述代码。
- 插件主页:plugins.jetbrains.com/plugin/9360…
- GitHub:github.com/gejun123456…
演示如下图:
只需安装插件,然后按Alt + Enter(macOS则是Option + Enter),即可自动生成对象转换代码。
方法二、借助对象映射框架实现对象转换
方法一虽然很方便,但如果对象的字段非常多,那么还是会导致代码非常啰嗦,不够简洁。
事实上,Java生态有很多对象映射框架,专门帮助我们实现对象间的转换。这里笔者列出了业界相对常用的选项:
产品 | Dozer | Orika | MapStruct | CGLib BeanCopier | Spring BeanUtils | Apache BeanUtils |
GitHub | dozer 1.9K stars | orika 1.2K stars | mapstruct 5K stars | cglib 4.3K stars | - | commons-beanutils 0.2K stars |
工作原理 | 大量反射,主要基于Field.set(obj, obj)为field赋值 | 基于javassist生成对象映射字节码,并加载生成的字节码文件 | 基于JSR269,在在编译期生成对象映射代码 | 基于ASM的MethodVisitor为field赋值 | 基于Spring反射工具类 | 基于反射 |
性能排名 | 5 | 2 | 1 | 4 | 3 | 6 |
虽然选项很多,但笔者目前只建议大家使用MapStruct。
MapStruct优势:
- 编译器生成Getter/Setter,无运行期性能损耗,性能强劲
- 基于JSR269,配置灵活
- 基于Getter/Setter,和自己手写Getter/Setter没有区别,搜索字段引用等较方便
缺点:
- 由于配置灵活,所以上手成本比其他组件稍微高一点点
MapStruct上手
配置IDE
参考 mapstruct.org/documentati… ,配置你的IDE
整合
- 在项目中添加如下内容:
<!-- ref: https://mapstruct.org/documentation/installation/ -->
<properties>
<org.mapstruct.version>1.4.2.Final</org.mapstruct.version>
</properties>
...
<dependencies>
<dependency>
<groupId>org.mapstruct</groupId>
<artifactId>mapstruct</artifactId>
<version>${org.mapstruct.version}</version>
</dependency>
</dependencies>
...
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.8.1</version>
<configuration>
<source>1.8</source> <!-- depending on your project -->
<target>1.8</target> <!-- depending on your project -->
<annotationProcessorPaths>
<path>
<groupId>org.mapstruct</groupId>
<artifactId>mapstruct-processor</artifactId>
<version>${org.mapstruct.version}</version>
</path>
<!-- other annotation processors -->
</annotationProcessorPaths>
</configuration>
</plugin>
</plugins>
</build>
- 如果你的项目使用了Lombok,或使用了spring-boot-configuration-processor,则使用类似如下的配置:
<properties>
<org.mapstruct.version>1.4.2.Final</org.mapstruct.version>
</properties>
...
<dependencies>
<dependency>
<groupId>org.mapstruct</groupId>
<artifactId>mapstruct</artifactId>
<version>${org.mapstruct.version}</version>
</dependency>
</dependencies>
...
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.8.1</version>
<configuration>
<source>8</source>
<target>8</target>
<encoding>UTF-8</encoding>
<!-- https://mapstruct.org/documentation/installation/ -->
<!-- https://mapstruct.org/documentation/stable/reference/html/#lombok -->
<annotationProcessorPaths>
<path>
<groupId>org.mapstruct</groupId>
<artifactId>mapstruct-processor</artifactId>
<version>${org.mapstruct.version}</version>
</path>
<path>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.18.16</version>
</path>
<path>
<groupId>org.projectlombok</groupId>
<artifactId>lombok-mapstruct-binding</artifactId>
<version>0.1.0</version>
</path>
<path>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-configuration-processor</artifactId>
<version>2.4.1</version>
</path>
</annotationProcessorPaths>
</configuration>
</plugin>
</plugins>
</build>
使用
定义接口,代码类似如下:
import com.itmuch.gogolive1.domain.User;
import com.itmuch.gogolive1.domain.UserRegisterReq;
import org.mapstruct.Mapper;
import org.mapstruct.factory.Mappers;
@Mapper
public interface UserConverter {
/**
* 固定写法:Mappers.getMapper(接口名.class);
*/
UserConverter INSTANCE = Mappers.getMapper(UserConverter.class);
User toUser(UserRegisterReq req);
}
使用:
@PostMapping("/users/reg")
public void reg2(@RequestBody UserRegisterReq userRegisterReq) {
// 省略password 与 confirmPassword等值判断
User user = UserConverter.INSTANCE.toUser(userRegisterReq);
// 保存user...
}
由代码可知,只需如下代码,即可将UserRegisterReq转换User。
User user = UserConverter.INSTANCE.toUser(userRegisterReq);
原理
编译代码,并在前面的UserConverter接口上,按快捷键 Command + Option + B (或点击 Navigate - Implementation(s) ) ,查找UserConverter的实现类,可跳转到类似如下代码:
@Generated(
value = "org.mapstruct.ap.MappingProcessor",
date = "2022-03-22T00:29:49+0800",
comments = "version: 1.4.2.Final, compiler: javac, environment: Java 11.0.9.1 (Azul Systems, Inc.)"
)
public class UserConverterImpl implements UserConverter {
@Override
public User toUser(UserRegisterReq req) {
if ( req == null ) {
return null;
}
UserBuilder user = User.builder();
user.email( req.getEmail() );
user.username( req.getUsername() );
user.password( req.getPassword() );
return user.build();
}
}
由代码可知,MapStruct在编译期间,生成了UserConverterImpl,并在其中实现了对象之间的转换。
和Spring整合
MapStruct支持与Spring整合,只需按如下步骤操作即可:
- 编写Mapper接口,并在其中添加 (componentModel = "spring") 属性:
@Mapper(componentModel = "spring")
public interface UserSpringConverter {
User toUser(UserRegisterReq req);
}
- 当使用时,只需注入 UserSpringConverter 即可:
@Autowired
UserSpringConverter userSpringConverter;
- 这是因为,使用 (componentModel = "spring") 后,生成的实现类会自动添加 @Component 注解
拓展
本文只是介绍了较为简单的例子,事实上,MapStruct支持非常灵活的配置,例如:
枚举映射
@Mapper
public interface OrderMapper {
OrderMapper INSTANCE = Mappers.getMapper( OrderMapper.class );
@ValueMappings({
@ValueMapping(target = "SPECIAL", source = "EXTRA"),
@ValueMapping(target = "DEFAULT", source = "STANDARD"),
@ValueMapping(target = "DEFAULT", source = "NORMAL")
})
ExternalOrderType orderTypeToExternalOrderType(OrderType orderType);
}
自定义表达式映射:
@Mapper
public interface SourceTargetMapper {
SourceTargetMapper INSTANCE = Mappers.getMapper( SourceTargetMapper.class );
@Mapping(target = "timeAndFormat",
expression = "java( new org.sample.TimeAndFormat( s.getTime(), s.getFormat() ) )")
Target sourceToTarget(Source s);
}
姿势繁多,有数十种,限于篇幅,无法一一枚举。建议大家移步 mapstruct.org/documentati… 查阅。个人经验来说,当转换的对象之间差异较大时,MapStruct的配置也会变得非常复杂,此时代码可读性、复杂性等等都比较高一些。个人不太喜欢折腾MapStruct,所以当遇到复杂情况时,笔者就直接手动实现对象映射了。毕竟,引入工具是帮助我们提效的,如果MapStruct需要大量配置,那意义就不大了。
作者:飞书技术
链接:https://juejin.cn/post/7099734198453272607
相关推荐
- YAML配置文件简介及使用(yaml 配置)
-
简介YAML是"YAMLAin'taMarkupLanguage"(YAML不是一种标记语言)的缩写。相比JSON格式的方便。...
- 教你如何解决最常见的58种网络故障排除方法
-
1.故障现象:网络适配器(网卡)设置与计算机资源有冲突。分析、排除:通过调整网卡资源中的IRQ和I/O值来避开与计算机其它资源的冲突。有些情况还需要通过设置主板的跳线来调整与其它资源的冲突。2.故障现...
- 一分钟带你了解服务器网卡(服务器网卡怎么用)
-
今天小编和大家聊一下服务器的网卡。什么是网卡?简单说网卡就是计算机与局域网互连的设备。计算机主要通过网卡接入网络。网卡又称为网络适配器或网络接口卡NIC(NetworkinterfaceCard)...
- linux文件之ssh配置文件的含义与作用
-
ssh远程登录命令是操作系统(包括linux和window系统)下常用的操作命令,可以帮助用户,远程登录服务器系统,查看,操作系统相关信息。linux系统对于ssh命令有专门保存其相关配置的目录和文件...
- Cilium 官方文档翻译 - IPAM(二)Kubernetes Host模式
-
KubernetesHostScopeciliumIPAM的kuberneteshost-scope模式通过选项ipam:kubernetes开启,将集群IP地址分配委托给每个独立的节点,并...
- 域名劫持跳转,域名劫持跳转的解决办法只需5步
-
简单来说,域名劫持就是把原本准备访问某网站的用户,在不知不觉中,劫持到仿冒的网站上,例如用户准备访问某家知名品牌的网上商店,黑客就可以通过域名劫持的手段,把其带到假的网上商店,同时收集用户的ID信息和...
- Linux 磁盘和文件系统管理(linux磁盘管理fdisk)
-
1检测并确认新硬盘...
- windows host文件怎么恢复?局域网访问全靠这些!
-
windowshost文件怎么恢复?windowshost文件是常用网址域名及其相应IP地址建立一个关联文件,通过这个host文件配置域名和IP的映射关系,以提高域名解析的速度,方便局域网用户使用...
- Nginx配置文件详解与优化建议(nginx 配置详解)
-
1、概述今天来详解一下Nginx的配置文件,以及给出一些配置建议,希望能对大家有所帮助。...
- Mac电脑hosts文件锁定,如何修改hosts文件权限
-
有时候我们需要修改hosts文件,但是网上很多教程都行不通,使用sudo命令也不行。其实有一个很简单的方法。打开终端命令行,使用如下命令即可:sudochflags-hvnoschg/etc/...
- windows电脑如何修改hosts文件?(windows 修改hosts文件)
-
先来简单说下电脑host的作用hosts文件的作用:hosts文件是一个用于储存计算机网络中各节点信息的计算机文件;作用是将一些常用的网址域名与其对应的IP地址建立一个关联“数据库”,当用户在浏览器中...
- Vigilante恶意软件行为怪异:修改Hosts文件以阻止受害者访问盗版网站
-
Sophos刚刚报道了一款名叫Vigilante的恶意软件,但其行为却让许多受害者感到不解。与其它专注于偷密码、搞破坏、或勒索赎金的恶意软件不同,Vigilante会通过修改Hosts文件...
- hosts文件无法修改几种现象和解决方法
-
第一种、hosts文件修改完不是直接保存而是弹出另存为窗口解决:1、右击hosts文件——属性——把“只读”前面勾去掉。第二种、打开hosts文件时提示“你没有权限打开该文件,请向文件的所有者或管理员...
- hosts文件位置在哪里,教你hosts文件位置在哪里
-
Hosts是一个没有扩展名的系统文件,其基本作用就是将一些常用的网址域名与其对应的IP地址建立一个关联"数据库",当用户在浏览器中输入一个需要登录的网址时,系统会首先自动从Hosts文件中寻找对应的I...
你 发表评论:
欢迎- 一周热门
-
-
前端面试:iframe 的优缺点? iframe有那些缺点
-
带斜线的表头制作好了,如何填充内容?这几种方法你更喜欢哪个?
-
漫学笔记之PHP.ini常用的配置信息
-
其实模版网站在开发工作中很重要,推荐几个参考站给大家
-
推荐7个模板代码和其他游戏源码下载的网址
-
[干货] JAVA - JVM - 2 内存两分 [干货]+java+-+jvm+-+2+内存两分吗
-
正在学习使用python搭建自动化测试框架?这个系统包你可能会用到
-
织梦(Dedecms)建站教程 织梦建站详细步骤
-
2024PHP在线客服系统源码+完全开源 带详细搭建教程
-
【开源分享】2024在线客服系统PHP源码(安装教程+全新UI)
-
- 最近发表
- 标签列表
-
- mybatis plus (70)
- scheduledtask (71)
- css滚动条 (60)
- java学生成绩管理系统 (59)
- 结构体数组 (69)
- databasemetadata (64)
- javastatic (68)
- jsp实用教程 (53)
- fontawesome (57)
- widget开发 (57)
- vb net教程 (62)
- hibernate 教程 (63)
- case语句 (57)
- svn连接 (74)
- directoryindex (69)
- session timeout (58)
- textbox换行 (67)
- extension_dir (64)
- linearlayout (58)
- vba高级教程 (75)
- iframe用法 (58)
- sqlparameter (59)
- trim函数 (59)
- flex布局 (63)
- contextloaderlistener (56)