百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

面试官:对象映射的方式对比与选择?

yuyutoo 2024-10-29 17:28 6 浏览 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

相关推荐

当 Linux 根分区 (/) 已满时如何释放空间?

根分区(/)是Linux文件系统的核心,包含操作系统核心文件、配置文件、日志文件、缓存和用户数据等。当根分区满载时,系统可能出现无法写入新文件、应用程序崩溃甚至无法启动的情况。常见原因包括:...

玩转 Linux 之:磁盘分区、挂载知多少?

今天来聊聊linux下磁盘分区、挂载的问题,篇幅所限,不会聊的太底层,纯当科普!!1、Linux分区简介1.1主分区vs扩展分区硬盘分区表中最多能存储四个分区,但我们实际使用时一般只分为两...

Linux 文件搜索神器 find 实战详解,建议收藏

在Linux系统使用中,作为一个管理员,我希望能查找系统中所有的大小超过200M文件,查看近7天系统中哪些文件被修改过,找出所有子目录中的可执行文件,这些任务需求...

Linux 操作系统磁盘操作(linux 磁盘命令)

一、文档介绍本文档描述Linux操作系统下多种场景下的磁盘操作情况。二、名词解释...

Win10新版19603推送:一键清理磁盘空间、首次集成Linux文件管理器

继上周四的Build19592后,微软今晨面向快速通道的Insider会员推送Windows10新预览版,操作系统版本号Build19603。除了一些常规修复,本次更新还带了不少新功能,一起来了...

Android 16允许Linux终端使用手机全部存储空间

IT之家4月20日消息,谷歌Pixel手机正朝着成为强大便携式计算设备的目标迈进。2025年3月的更新中,Linux终端应用的推出为这一转变奠定了重要基础。该应用允许兼容的安卓设备...

Linux 系统管理大容量磁盘(2TB+)操作指南

对于容量超过2TB的磁盘,传统MBR分区表的32位寻址机制存在限制(最大支持2.2TB)。需采用GPT(GUIDPartitionTable)分区方案,其支持64位寻址,理论上限为9.4ZB(9....

Linux 服务器上查看磁盘类型的方法

方法1:使用lsblk命令lsblk输出说明:TYPE列显示设备类型,如disk(物理磁盘)、part(分区)、rom(只读存储)等。...

ESXI7虚机上的Ubuntu Linux 22.04 LVM空间扩容操作记录

本人在实际的使用中经常遇到Vmware上安装的Linux虚机的LVM扩容情况,最终实现lv的扩容,大多数情况因为虚机都是有备用或者可停机的情况,一般情况下通过添加一块物理盘再加入vg,然后扩容lv来实...

5.4K Star很容易!Windows读取Linux磁盘格式工具

[开源日记],分享10k+Star的优质开源项目...

Linux 文件系统监控:用脚本自动化磁盘空间管理

在Linux系统中,文件系统监控是一项非常重要的任务,它可以帮助我们及时发现磁盘空间不足的问题,避免因磁盘满而导致的系统服务不可用。通过编写脚本自动化磁盘空间管理,我们可以更加高效地处理这一问题。下面...

Linux磁盘管理LVM实战(linux实验磁盘管理)

LVM(逻辑卷管理器,LogicalVolumeManager)是一种在Linux系统中用于灵活管理磁盘空间的技术,通过将物理磁盘抽象为逻辑卷,实现动态调整存储容量、跨磁盘扩展等功能。本章节...

Linux查看文件大小:`ls`和`du`为何结果不同?一文讲透原理!

Linux查看文件大小:ls和du为何结果不同?一文讲透原理!在Linux运维中,查看文件大小是日常高频操作。但你是否遇到过以下困惑?...

使用 df 命令检查服务器磁盘满了,但用 du 命令发现实际小于磁盘容量

在Linux系统中,管理员或开发者经常会遇到一个令人困惑的问题:使用...

Linux磁盘爆满紧急救援指南:5步清理释放50GB+小白也能轻松搞定

“服务器卡死?网站崩溃?当Linux系统弹出‘Nospaceleft’的红色警报,别慌!本文手把手教你从‘删库到跑路’进阶为‘磁盘清理大师’,5个关键步骤+30条救命命令,快速释放磁盘空间,拯救你...

取消回复欢迎 发表评论: