实体映射解决方案-MapStruct

序言

本文给大家介绍 MapStruct。

一、问题引入

在我们的日常开发过程中,我们经常会将 VO、DTO、PO …… 等对象相互进行映射。实现的方式可能有以下几种:

  1. 自己手动进行转换
  2. 利用诸如 Spring 或 Apache 提供的 BeanUtils 等工具类

如果自己手动进行转换,这将会增加开发人员的工作量。如果利用上述的这些工具类通常会存在性能问题(因为这些工具类通常使用的是反射机制)。

二、MapStruct

image.png

MapStruct 是一个用于生成类型安全的 bean 映射类的 Java 注解处理器。开发人员通过定义一个映射器接口,该接口声明任何所需的映射方法。在编译过程中,MapStruct 将生成此接口的实现

三、快速入门

3.1 导入依赖

<dependency><groupId>org.mapstruct</groupId><artifactId>mapstruct</artifactId><version>1.5.5.Final</version>
</dependency><!--此依赖必须位于 Lombok 依赖的下面-->
<dependency><groupId>org.mapstruct</groupId><artifactId>mapstruct-processor</artifactId><version>1.5.5.Final</version>
</dependency>

3.2 定义接口

@Mapper
public interface UserMapper {// 固定写法UserMapper INSTANCE = Mappers.getMapper(UserMapper.class);// 参数是 User 实体类,返回 UserDTOUserDTO userToUserDTO(User user);
}

3.3 使用

@Test
public void test() {User user = new User(1, "zs");// 直接调用定义好的接口方法// User 实体类转换成 UserDTOUserDTO userDTO = UserMapper.INSTANCE.userToUserDTO(user);System.out.println(userDTO);
}

测试效果:

image.png

四、映射器

在快速入门的例子中,User 和 UserDTO 的字段都是相对应的。如果需要映射字段是不相同的呢?MapperStruct 提供了映射器可以帮助我们解决这些问题。

4.1 基本映射

@Data
@NoArgsConstructor
@AllArgsConstructor
public class User {private int id;private String name;
}@Data
@NoArgsConstructor
@AllArgsConstructor
public class UserDTO {private int id;private String username;
}

如果我们需要将 User 的 name 字段映射到 username,只需要按如下方式简单的添加一个注解就行了。

@Mapper
public interface UserMapper {UserMapper INSTANCE = Mappers.getMapper(UserMapper.class);// 使用 @Mapping 表示映射关系@Mapping(target = "username", source = "name")UserDTO userToUserDTO(User user);
}

4.2 组合注解

为了能够让开发者更加方便的使用,MapStrut 提供了自定义组合注解的方式。

@Data
@NoArgsConstructor
@AllArgsConstructor
public class User {private int id;private String name;
}@Data
@NoArgsConstructor
@AllArgsConstructor
public class UserDTO {private int id;private String name;private String username;
}

如果我们想要将 User 的 name 都映射到 UserDTO 的 name 和 username,我们可以采用以下的方式:

  1. 定义一个组合注解

    	@Retention(RetentionPolicy.CLASS)@Mapping(target = "username", source = "name")public @interface Composition {}
    
  2. 使用组合注解

    	@Mapperpublic interface UserMapper {UserMapper INSTANCE = Mappers.getMapper(UserMapper.class);@CompositionUserDTO userToUserDTO(User user);}
    

    组合注解可以达到与直接使用 @Mapping 注解相同的效果:

    image.png

五、映射方法

@Data
@NoArgsConstructor
@AllArgsConstructor
public class User {private int id;private String name;
}@Data
@NoArgsConstructor
@AllArgsConstructor
public class UserDTO {private int id;private String name;private int nameLen;
}

在开发的过程中,有时映射逻辑可能比较复杂。MapStruct 提供的注解映射器可能无法满足。例如:我们需要在映射时将上面 name 的长度映射到 nameLen。
MapStruct 可以使用接口默认方法解决特殊的映射需求。

@Mapper
public interface UserMapper {UserMapper INSTANCE = Mappers.getMapper(UserMapper.class);default UserDTO userToUserDTO(User user) {UserDTO userDTO = new UserDTO();userDTO.setId(user.getId());userDTO.setName(user.getName());userDTO.setNameLen(user.getName().length());return userDTO;}
}

测试效果:

image.png

六、装饰器类

上面的映射方法其实就是手动实现映射关系。但是,开发人员可能只想修改几个特殊的字段映射,这时可以利用装饰器类来处理映射。

@Mapper
// 使用装饰器
@DecoratedWith(UserMapperDecorator.class)
public interface UserMapper {UserMapper INSTANCE = Mappers.getMapper(UserMapper.class);UserDTO userToUserDTO(User user);}// 装饰器类
public abstract class UserMapperDecorator implements UserMapper {private final UserMapper delegate;public UserMapperDecorator(UserMapper delegate) {this.delegate = delegate;}@Overridepublic UserDTO userToUserDTO(User user) {UserDTO userDTO = delegate.userToUserDTO(user);// 特殊处理 nameLen 字段映射userDTO.setNameLen(user.getName().length());return userDTO;}
}

七、依赖注入

现在,我们项目的开发一般都是采用的 Spring。MapStruct 提供了更加适合 Spring 的依赖注入方式。

// componentModel 指定使用 Spring 模式
@Mapper(componentModel = MappingConstants.ComponentModel.SPRING)
@DecoratedWith(UserMapperDecorator.class)
public interface UserMapper {UserDTO userToUserDTO(User user);
}public abstract class UserMapperDecorator implements UserMapper {// 装饰器中采用注解获取@Resourceprivate UserMapper delegate;@Overridepublic UserDTO userToUserDTO(User user) {UserDTO userDTO = delegate.userToUserDTO(user);userDTO.setNameLen(user.getName().length());return userDTO;}
}

使用方式:

// 使用注解注入
@Resource
private UserMapper userMapper;@Test
public void test() {User user = new User(1, "zs");UserDTO userDTO = userMapper.userToUserDTO(user);System.out.println(userDTO);
}

八、类型转换

在实体类映射的过程中,我们经常会遇到如下的问题:

@Data
@NoArgsConstructor
@AllArgsConstructor
public class UserDTO {private int id;private String name;// 在 dto 中 status 字段是 boolean 类型private boolean status;
}@Data
@NoArgsConstructor
@AllArgsConstructor
public class UserPO {private int id;private String name;// 在 po 中 status 字段是 status 类型private Integer status;
}

如果需要类型转换,可以通过以下方式处理:

// 新增一个类型映射
// 如果使用了 Spring 模式,则需要注入
@Component
public class IntBooleanMapper {public boolean intToBoolean(int value) {return value == 1;}public int booleanToInt(boolean value) {return value ? 1 : 0;}
}// 使用 uses 指定
@Mapper(componentModel = MappingConstants.ComponentModel.SPRING,uses = {IntBooleanMapper.class})
@DecoratedWith(UserMapperDecorator.class)
public interface UserMapper {UserDTO userToUserDTO(User user);UserDTO UserPOToUserDTO(UserPO userPO);}

测试类型转换:

@Resource
private UserMapper userMapper;@Test
public void test() {UserPO userPO = new UserPO(1, "ZS", 1);UserDTO userDTO = userMapper.UserPOToUserDTO(userPO);System.out.println(userDTO);
}

测试效果:

image.png

九、默认值

在 MapStruct 中,还可以指定映射的默认值。

@Mapper(componentModel = MappingConstants.ComponentModel.SPRING)
public interface UserMapper {// 使用 defaultValue 定义默认值为 default// 如果 User 的 name 为 null,那么 UserDTO 的 name 值则为 default@Mapping(target = "name", defaultValue = "default")UserDTO userToUserDTO(User user);}

十、FAQ

  1. MapStruct 依赖与 Lombok 依赖可能存在冲突的情况,在引入 MapStruct 相关依赖时要严格安装本文提供的顺序引入。
  2. MapStruct 的原理是通过定义的接口,自动帮助开发人员生成映射代码。所以当修改了接口中内容,若没有立即生效则可能需要使用 mvn cleanmvn compile 等命令清理一下自动生成的旧版本代码。

往期推荐

  1. 动态切换数据源的最佳实践
  2. Gateway
  3. 缓存神器-JetCache
  4. Mybatis 缓存机制
  5. 为什么 MySQL 单表数据量最好别超过 2000w
  6. IoC 思想简单而深邃
  7. ThreadLocal

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/1410440.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

如何搭建本地的 NPM 私有仓库 Nexus

NPM 本地私有仓库&#xff0c;是在本地搭建NPM私有仓库&#xff0c;对公司级别的组件库进行管理。在日常开发中&#xff0c;经常会遇到抽象公共组件的场景&#xff0c;在项目内部进行公用。新的项目开始时&#xff0c;也会拷贝一份创建一个新的项目&#xff0c;这样做不易于管理…

单片机编程实例400例大全(100-200)

今天继续分享单片机编程实例第100-200例。 今天的实例会比前面100复杂一些&#xff0c;我大概看了下&#xff0c;很多都具备实际产品的参考价值。 今天继续分享单片机编程实例第100-200例。 今天的实例会比前面100复杂一些&#xff0c;我大概看了下&#xff0c;很多都具备实际…

gitlab设置保护分支

gitlab设置保护分支方法 进入代码仓库首页&#xff0c;找到settings下的repository并点击进入 找到Protected Branches 下的Exoand按钮&#xff0c;并点击展开 可以看到已经存在默认的保护分支&#xff0c;通常是master/main分支&#xff0c;也可以添加新的保护分支 新建保护分…

Mybatis高级

1. Mybatis多表查询概念 ​ 在学习多表查询的之前&#xff0c;我们要先明确多表的关系都有哪些&#xff0c;如何划分。 1.1 多表关系的划分 一对一 一对一的关系是两张表之间 A表的一条数据只能对应B表的一条数据。比如 订单表和用户表 一个订单只能属于…

Vue阶段练习:组件拆分

页面开发思路 分析页面&#xff0c;按模块拆分组件&#xff0c;搭架子&#xff08;局部或全局注册&#xff09;根据设计图&#xff0c;编写html结构css样式拆分封装通用小组件&#xff08;局部或全局注册&#xff09;将来通过js动态渲染实现功能 BaseBrandItem.vue <templ…

408数据结构-二叉树的遍历 自学知识点整理

前置知识&#xff1a;二叉树的概念、性质与存储结构 二叉树的遍历 二叉树的遍历是指按某条搜索路径访问树中每个结点&#xff0c;使得每个结点均被访问一次&#xff0c;而且仅被访问一次。 二叉树的递归特性: ①要么是棵空二叉树&#xff1b; ②要么就是由“根节点左子树右子树…

渗透之sql注入联合查询的注入

sql注入产生的原因&#xff1a; 由于程序过滤不严谨&#xff0c;导致用户有一些异常输入&#xff0c;最终触发数据库的查询。所以会出现sql注入这个问题。有些恶意的人就会利用这些信息导致数据库泄露。 注意&#xff1a;一般我们存在注入点我们会查询管理员的账号和密码&#…

Q1季度家用健身器械行业线上市场销售数据分析

自疫情开始&#xff0c;全民健身的浪潮就持续至今。然而&#xff0c;水能载舟亦能覆舟&#xff0c;一边是不断释放的健身需求&#xff0c;另一边却是无数健身房的闭店潮。 越来越多人倾向于选择家用健身器械来运动或是直接选择无器械的健身运动&#xff0c;比如各类健身操。而…

【跟马少平老师学AI】-【神经网络是怎么实现的】(六)过拟合问题

一句话归纳&#xff1a; 1&#xff09;过拟合问题&#xff1a; 图中的点为样本直线欠拟合曲线2过拟合 2&#xff09;迭代次数与拟合情况&#xff1a; 训练次数过多可能导致过拟合。 3&#xff09;正则化项法弱化过拟后问题&#xff1a; 加正则项&#xff0c;在最小化损失函数时…

电脑自带dll修复在哪里,使用dll修复工具解决dll问题

在我们日常与电脑相伴的工作与学习过程中&#xff0c;我们经常会遇到一些错误提示&#xff0c;其中最常见的就是“无法找到.dll”或“找不到.dll文件”。这种情况通常是由于dll文件丢失或损坏导致的。dll文件是动态链接库文件&#xff0c;它包含了许多程序运行所需的函数和资源…

场景文本检测识别学习 day06(Vi-Transformer论文精读、MAE论文阅读)

Vi-Transformer论文精读 在NLP领域&#xff0c;基于注意力的Transformer模型使用的非常广泛&#xff0c;但是在计算机视觉领域&#xff0c;注意力更多是和CNN一起使用&#xff0c;或者是单纯将CNN的卷积替换成注意力&#xff0c;但是整体的CNN 架构没有发生改变VIT说明&#x…

B树:原理、操作及应用

B树&#xff1a;原理、操作及应用 一、引言二、B树概述1. 定义与性质2. B树与磁盘I/O 三、B树的基本操作1. 搜索&#xff08;B-TREE-SEARCH&#xff09;2. 插入&#xff08;B-TREE-INSERT&#xff09;3. 删除&#xff08;B-TREE-DELETE&#xff09; 四、B树的C代码实现示例五、…

一文全面了解 wxWidgets 布局器(Sizers)

目录 Sizers背后的理念 共同特征 最小大小 边框 对齐方式 伸缩因子 使用 Sizer 隐藏控件 wxBoxSizer wxStaticBoxSizer wxGridSizer wxFlexGridSizer 布局器&#xff08;Sizers&#xff09;&#xff0c;由wxWidgets类层次结构中的wxSizer类及其派生类表示&#xff0…

基于 Wireshark 分析 IP 协议

一、IP 协议 IP&#xff08;Internet Protocol&#xff09;协议是一种网络层协议&#xff0c;它用于在计算机网络中实现数据包的传输和路由。 IP协议的主要功能有&#xff1a; 1. 数据报格式&#xff1a;IP协议将待传输的数据分割成一个个数据包&#xff0c;每个数据包包含有…

android init进程启动流程

Android系统完整的启动流程 android 系统架构图 init进程的启动流程 init进程启动服务的顺序 bool Service::Start() {// Starting a service removes it from the disabled or reset state and// immediately takes it out of the restarting state if it was in there.flags_…

JAVA面试专题-微服务篇

Spring cloud Spring Cloud 5大组件有哪些 注册中心/配置中心&#xff1a;nacos 负载均衡&#xff1a;Ribbon 服务远程调用&#xff1a;Feign 服务保护&#xff1a;sentinel 服务网关&#xff1a;Gateway 微服务注册和发现 nacos和eureka的区别 负载均衡 微服务向Ribbon发送…

[1678]旅游景点信息Myeclipse开发mysql数据库web结构java编程计算机网页项目

一、源码特点 JSP 旅游景点信息管理系统是一套完善的java web信息管理系统&#xff0c;对理解JSP java编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用B/S模式开发。开发环境为TOMCAT7.0,Myeclipse8.5开发&#xff0c;数据库为Mysql…

Word页脚设置“第X页共X页”的方法【域实现】

Word页脚设置“第X页共X页”的方法【域实现】 在设置Word页码格式的要求中&#xff0c;有时需要设置为“第X页共X页”这种格式&#xff0c;使用Word中的域功能可实现&#xff0c;同时&#xff0c;在某些情况下&#xff0c;可能还需要减去封面的页码&#xff0c;接下来为具体步…

软件标准建设体系规范过程性文档(软件开发,管理,安全,运维等各阶段全文档)

软件标准建设体系规范是确保软件开发过程标准化、高质量和可维护性的关键。它通常包括一系列文档、规范、流程和最佳实践&#xff0c;以确保软件项目的成功实施和交付。以下是一个软件标准建设体系规范的基本框架&#xff1a; 软件全套资料获取方式1&#xff1a;进主页。 获取…

C#描述-计算机视觉OpenCV(3):重映射

C#描述-计算机视觉OpenCV&#xff08;3&#xff09;&#xff1a;重映射 前言色彩波形图像重映射 前言 C#描述-计算机视觉OpenCV&#xff08;1&#xff09;&#xff1a;基础操作 C#描述-计算机视觉OpenCV&#xff08;2&#xff09;&#xff1a;图像处理 在前文中&#xff0c;描…