简单尝试了下发现比Dozer还有BeanUtil还方便小巧

注解的作用是在生成字节码文件时实现具体GetterSetter方法,实际转换时就是赋值操作,嘎嘎快

参考文章:

https://juejin.cn/post/7140149801991012365

  

引入必须的依赖:

lombok一般项目都会添加,这里我就只放这两个

<dependency>
<groupId>org.mapstruct</groupId>
<artifactId>mapstruct</artifactId>
<version>1.5.0.Final</version>
</dependency>
<dependency>
<groupId>org.mapstruct</groupId>
<artifactId>mapstruct-processor</artifactId>
<version>1.5.0.Final</version>
</dependency>

 

一、入门案例:

两个需要相互转换的实体类,演示用:

DTO

package cn.cloud9.server.test.mapstruct;

import lombok.*;
import lombok.experimental.Accessors; @Data
@Builder
@ToString
@Accessors(chain = true)
@AllArgsConstructor
@NoArgsConstructor
public class DemoDTO {
private Integer fieldA;
private Boolean fieldB;
private String fieldC;
}

Entity

package cn.cloud9.server.test.mapstruct;

import lombok.*;
import lombok.experimental.Accessors; @Data
@Builder
@ToString
@Accessors(chain = true)
@AllArgsConstructor
@NoArgsConstructor
public class DemoEntity {
private Integer fieldD;
private Boolean fieldE;
private String fieldF;
}

  

转换器接口编写:

如果需要转换更多的实体类,都可以在接口声明方法,然后声明映射的字段

package cn.cloud9.server.test.mapstruct;

import org.mapstruct.Mapper;
import org.mapstruct.Mapping;
import org.mapstruct.Mappings;
import org.mapstruct.factory.Mappers; /**
* 转换器接口
*/
@Mapper
public interface DemoConverter {
DemoConverter INSTANCE = Mappers.getMapper(DemoConverter.class); /* DTO -> ENTITY */
@Mappings(value = {
@Mapping(source = "fieldA", target = "fieldD"),
@Mapping(source = "fieldB", target = "fieldE"),
@Mapping(source = "fieldC", target = "fieldF"),
})
DemoEntity dto2entity(DemoDTO demoDTO); /* ENTITY -> DTO */
@Mappings(value = {
@Mapping(source = "fieldD", target = "fieldA"),
@Mapping(source = "fieldE", target = "fieldB"),
@Mapping(source = "fieldF", target = "fieldC"),
})
DemoDTO entity2dto(DemoEntity demoEntity);
}

  

测试方法:

    @Test
public void converterTest() {
DemoDTO build = DemoDTO.builder()
.fieldA(1001)
.fieldB(Boolean.TRUE)
.fieldC("TEST")
.build(); DemoEntity demoEntity = DemoConverter.INSTANCE.dto2entity(build);
DemoDTO demoDTO = DemoConverter.INSTANCE.entity2dto(demoEntity);
log.info("demoEntity -> {}", demoEntity);
log.info("demoDTO -> {}", demoDTO);
}

  

执行结果:

22:52:20.079 [main] INFO cn.cloud9.spring.BeanFactoryTest - demoEntity -> DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST)
22:52:20.083 [main] INFO cn.cloud9.spring.BeanFactoryTest - demoDTO -> DemoDTO(fieldA=1001, fieldB=true, fieldC=TEST) Process finished with exit code 0

 

二、集合转换支持

也是根据参数类型推断,自动生成了迭代集合

    /* List<ENTITY -> DTO>  */
@Mappings(value = {
@Mapping(source = "fieldD", target = "fieldA"),
@Mapping(source = "fieldE", target = "fieldB"),
@Mapping(source = "fieldF", target = "fieldC"),
})
List<DemoDTO> entity2dto(List<DemoEntity> demoEntity); /* List<DTO -> ENTITY> */
@Mappings(value = {
@Mapping(source = "fieldA", target = "fieldD"),
@Mapping(source = "fieldB", target = "fieldE"),
@Mapping(source = "fieldC", target = "fieldF"),
})
List<DemoEntity> dto2entity(List<DemoDTO> demoDTO);

测试方法:

ArrayList<DemoDTO> objects = new ArrayList<>();
for (int i = 0; i < 10; i++) {
objects.add(DemoDTO.builder()
.fieldA(1001)
.fieldB(Boolean.TRUE)
.fieldC("TEST")
.build());
}
List<DemoEntity> demoEntityList = DemoConverter.INSTANCE.dto2entity(objects);
demoEntityList.forEach(d -> log.info("de {}", d));

执行结果:

23:16:13.845 [main] INFO cn.cloud9.spring.BeanFactoryTest - de DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST)
23:16:13.845 [main] INFO cn.cloud9.spring.BeanFactoryTest - de DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST)
23:16:13.845 [main] INFO cn.cloud9.spring.BeanFactoryTest - de DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST)
23:16:13.845 [main] INFO cn.cloud9.spring.BeanFactoryTest - de DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST)
23:16:13.845 [main] INFO cn.cloud9.spring.BeanFactoryTest - de DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST)
23:16:13.845 [main] INFO cn.cloud9.spring.BeanFactoryTest - de DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST)
23:16:13.845 [main] INFO cn.cloud9.spring.BeanFactoryTest - de DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST)
23:16:13.845 [main] INFO cn.cloud9.spring.BeanFactoryTest - de DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST)
23:16:13.845 [main] INFO cn.cloud9.spring.BeanFactoryTest - de DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST)
23:16:13.845 [main] INFO cn.cloud9.spring.BeanFactoryTest - de DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST)

  

三、映射配置继承与反向继承

只需要提供一份映射注解信息即可,其他方法通过继承注解获取映射信息

同理,要反向转换时,提供了一个反向注解:

两个注解都需要你提供方法名来查找映射配置

package cn.cloud9.server.test.mapstruct;

import org.mapstruct.*;
import org.mapstruct.factory.Mappers; import java.util.List; /**
* 转换器接口
*/
@Mapper
public interface DemoConverter {
DemoConverter INSTANCE = Mappers.getMapper(DemoConverter.class); /* DTO -> ENTITY */
@Mappings(value = {
@Mapping(source = "fieldA", target = "fieldD"),
@Mapping(source = "fieldB", target = "fieldE"),
@Mapping(source = "fieldC", target = "fieldF"),
})
DemoEntity dto2entity(DemoDTO demoDTO); /* ENTITY -> DTO */
// @Mappings(value = {
// @Mapping(source = "fieldD", target = "fieldA"),
// @Mapping(source = "fieldE", target = "fieldB"),
// @Mapping(source = "fieldF", target = "fieldC"),
// })
// DemoDTO entity2dto(DemoEntity demoEntity);
@InheritInverseConfiguration(name = "dto2entity")
DemoDTO entity2dto(DemoEntity demoEntity); /* List<ENTITY -> DTO> */
// @Mappings(value = {
// @Mapping(source = "fieldD", target = "fieldA"),
// @Mapping(source = "fieldE", target = "fieldB"),
// @Mapping(source = "fieldF", target = "fieldC"),
// })
@InheritConfiguration(name = "entity2dto")
List<DemoDTO> entity2dto(List<DemoEntity> demoEntity); /* List<DTO -> ENTITY> */
// @Mappings(value = {
// @Mapping(source = "fieldA", target = "fieldD"),
// @Mapping(source = "fieldB", target = "fieldE"),
// @Mapping(source = "fieldC", target = "fieldF"),
// })
@InheritInverseConfiguration(name = "entity2dto")
List<DemoEntity> dto2entity(List<DemoDTO> demoDTO);
}

  

四、是否阻止默认映射

默认同名同类型字段直接映射

这里对两个实体类新加了同一个字段:

private Long filedG;

 

测试时添加该属性:

@Test
public void converterTest() {
DemoDTO build = DemoDTO.builder()
.fieldA(1001)
.fieldB(Boolean.TRUE)
.fieldC("TEST")
.filedG(3003L)
.build();
DemoEntity demoEntity = DemoConverter.INSTANCE.dto2entity(build);
DemoDTO demoDTO = DemoConverter.INSTANCE.entity2dto(demoEntity);
log.info("demoEntity -> {}", demoEntity);
log.info("demoDTO -> {}", demoDTO);
}

执行结果可以发现,默认赋值上去了

23:34:51.780 [main] INFO cn.cloud9.spring.BeanFactoryTest - demoEntity -> DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST, filedG=3003)
23:34:51.787 [main] INFO cn.cloud9.spring.BeanFactoryTest - demoDTO -> DemoDTO(fieldA=1001, fieldB=true, fieldC=TEST, filedG=3003) Process finished with exit code 0

  

使用@BeanMapping注解可以阻止这种赋值行为:

/**
* 转换器接口
*/
@Mapper
public interface DemoConverter {
DemoConverter INSTANCE = Mappers.getMapper(DemoConverter.class); /* @BeanMapping(ignoreByDefault = true) 阻止MapStruct默认同名字段赋值行为 */
@BeanMapping(ignoreByDefault = true)
/* DTO -> ENTITY */
@Mappings(value = {
@Mapping(source = "fieldA", target = "fieldD"),
@Mapping(source = "fieldB", target = "fieldE"),
@Mapping(source = "fieldC", target = "fieldF"),
})
DemoEntity dto2entity(DemoDTO demoDTO); /* ENTITY -> DTO */
@InheritInverseConfiguration(name = "dto2entity")
DemoDTO entity2dto(DemoEntity demoEntity); /* List<ENTITY -> DTO> */
@InheritConfiguration(name = "entity2dto")
List<DemoDTO> entity2dto(List<DemoEntity> demoEntity); /* List<DTO -> ENTITY> */
@InheritInverseConfiguration(name = "entity2dto")
List<DemoEntity> dto2entity(List<DemoDTO> demoDTO);
}

  

再次执行发现不再能够进行赋值处理了:

23:38:03.011 [main] INFO cn.cloud9.spring.BeanFactoryTest - demoEntity -> DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST, filedG=null)
23:38:03.015 [main] INFO cn.cloud9.spring.BeanFactoryTest - demoDTO -> DemoDTO(fieldA=1001, fieldB=true, fieldC=TEST, filedG=null)

  

是否不继承@BeanMapping忽略默认映射的配置?

https://www.bilibili.com/video/BV1E5411n7HR?p=9

在视频中提到是不继承的,这里我加上了字段的赋值,并且调用反向转换的方法:

DemoDTO build = DemoDTO.builder()
.fieldA(1001)
.fieldB(Boolean.TRUE)
.fieldC("TEST")
.fieldG(3003L)
.build();
DemoEntity demoEntity = DemoConverter.INSTANCE.dto2entity(build);
demoEntity.setFieldG(3003L);
DemoDTO demoDTO = DemoConverter.INSTANCE.entity2dto(demoEntity);
log.info("demoEntity -> {}", demoEntity);
log.info("demoDTO -> {}", demoDTO);

可以从执行结果发现,实际上是没有赋值的,说明是继承了@BeanMapping的效果了

23:41:02.989 [main] INFO cn.cloud9.spring.BeanFactoryTest - demoEntity -> DemoEntity(fieldD=1001, fieldE=true, fieldF=TEST, fieldG=3003)
23:41:02.994 [main] INFO cn.cloud9.spring.BeanFactoryTest - demoDTO -> DemoDTO(fieldA=1001, fieldB=true, fieldC=TEST, fieldG=null) Process finished with exit code 0

  

五、注册成SpringBean

在前面的用法中都是直接声明一个单例使用,这里可以交给Spring处理:

@Mapper(componentModel = "spring")

package cn.cloud9.server.test.mapstruct;

import org.mapstruct.*;
import org.mapstruct.factory.Mappers; import java.util.List; /**
* 转换器接口
*/
@Mapper(componentModel = "spring")
public interface DemoConverter {
DemoConverter INSTANCE = Mappers.getMapper(DemoConverter.class); /* @BeanMapping(ignoreByDefault = true) 阻止MapStruct默认同名字段赋值行为 */
// @BeanMapping(ignoreByDefault = true)
/* DTO -> ENTITY */
@Mappings(value = {
@Mapping(source = "fieldA", target = "fieldD"),
@Mapping(source = "fieldB", target = "fieldE"),
@Mapping(source = "fieldC", target = "fieldF"),
})
DemoEntity dto2entity(DemoDTO demoDTO); /* ENTITY -> DTO */
@InheritInverseConfiguration(name = "dto2entity")
DemoDTO entity2dto(DemoEntity demoEntity); /* List<ENTITY -> DTO> */
@InheritConfiguration(name = "entity2dto")
List<DemoDTO> entity2dto(List<DemoEntity> demoEntity); /* List<DTO -> ENTITY> */
@InheritInverseConfiguration(name = "entity2dto")
List<DemoEntity> dto2entity(List<DemoDTO> demoDTO);
}

  

在需要调用的地方声明即可:

package cn.cloud9.server.test.controller;

import cn.cloud9.server.test.mapstruct.DemoConverter;
import cn.cloud9.server.test.mapstruct.DemoDTO;
import cn.cloud9.server.test.mapstruct.DemoEntity;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController; import javax.annotation.Resource; @RestController
@RequestMapping("/test/map-struct")
public class MapStructController { @Resource
private DemoConverter demoConverter; @GetMapping("/get")
public DemoEntity converterTest() {
DemoDTO build = DemoDTO.builder()
.fieldA(1001)
.fieldB(Boolean.TRUE)
.fieldC("TEST")
.fieldG(3003L)
.build();
return demoConverter.dto2entity(build);
}
}

测试结果:

  

【Java】实体类转换框架 MapStruct的更多相关文章

  1. Java实体映射工具MapStruct的使用

    官网地址:http://mapstruct.org/ MapStruct 是一个代码生成器,简化了不同的 Java Bean 之间映射的处理,所谓的映射指的就是从一个实体变化成一个实体.例如我们在实际 ...

  2. 利用在线工具根据JSon数据自动生成对应的Java实体类

    如果你希望根据JSon数据自动生成对应的Java实体类,并且希望能进行变量的重命名,那么“JSON To Java”一定适合你.(下面的地址需要FQ) https://jsontojava.appsp ...

  3. 我写了一个java实体类,implements了Serializable接口,然后我如何让serialversionUID自动生成

    写了一个java实体类,implements了Serializable接口,让serialversionUID自动生成方法: 1.点击类旁边的警告符号: 2.选择Add generated seria ...

  4. 利用JAXB实现java实体类和xml互相转换

    1.应用场景 在使用WebService实现数据上传下载,数据查询时,可以利用JAXB实现java实体类和xml互相转换 2.Demo 2.1 student.java 实体类,包含list(set同 ...

  5. MyEclipse 利用反向功能生成Java 实体类

    1.Window -> Open Perspective -> MyEclipse Database Explorer 到DB Broswer界面 2.右键 -> New,新建一个数 ...

  6. 在线数据库表(sql语句)生成java实体类工具

    相信每个做java开发的读者,都接触过SQL建表语句,尤其是在项目开发初期,因为数据库是项目的基石. 在现代项目开发中,出现了许多ORM框架,通过简单的实体映射,即可实现与数据库的交互,然而我们最初设 ...

  7. Mysql逆向工程效率神器之使用IDE自动生成Java实体类

    Mysql逆向工程效率神器之使用IDE自动生成Java实体类 简介:实战使用IDE根据Mysql自动生成java pojo实体类 1.IDEA连接数据库 菜单View→Tool Windows→Dat ...

  8. JAVA实体类不要使用基本类型,基本类型包含byte、int、short、long、float、double、char、boolean

    由于JAVA的基本类型会有默认值,例如当某个类中存在private  int age;字段时,创建这个类时,age会有默认值0.当使用age属性时,它总会有值.因此在某些情况下,便无法实现age为nu ...

  9. xml文档的解析并通过工具类实现java实体类的映射:XML工具-XmlUtil

    若有疑问,可以联系我本人微信:Y1141100952 声明:本文章为原稿,转载必须说明 本文章地址,否则一旦发现,必追究法律责任 1:本文章显示通过 XML工具-XmlUtil工具实现解析soap报文 ...

  10. 小D课堂-SpringBoot 2.x微信支付在线教育网站项目实战_2-6.Mysql逆向工程效率神器之使用IDE自动生成Java实体类

    笔记 6.Mysql逆向工程效率神器之使用IDE自动生成Java实体类     简介:实战使用IDE根据Mysql自动生成java pojo实体类                  1.IDEA连接数 ...

随机推荐

  1. itest(爱测试) 开源接口测试,敏捷测试管理平台10.1.0发布

    一:itest work 简介 itest work 开源敏捷测试管理,包含极简的任务管理,测试管理,缺陷管理,测试环境管理,接口测试,接口Mock,还有压测 ,又有丰富的统计分析,8合1工作站.可按 ...

  2. testArticle

    Test Article This is a test article for ArticleSync. Test Edit...... test Edit

  3. flutter 尝试创建第一个页面(三)

    新建目录 assets  存放图片 在pubspec..yaml 中添加 flutter: # The following line ensures that the Material Icons f ...

  4. hibernate映射对照表

    2.3. Basic Types Basic value types usually map a single database column, to a single, non-aggregated ...

  5. 【Playwright+Python】系列教程(一)环境搭建及脚本录制

    前言 看到这个文章,有的同学会说: 六哥,你为啥不早早就写完python系列的文章. 因为有徒弟需要吧,如果你也想学自学,那这篇文章,可以说是我们结缘一起学习的开始吧! 如果对你有用,建议收藏和转发! ...

  6. Jupyter QtConsole 配置,2023 年了你还在使用 QtConsole 吗?

    目录 Jupyter QtConsole 配置,2023 年了你还在使用 QtConsole 吗? Jupyter QtConsole 的安装 设置字体 启动时自动加载需要的库包 更新:2023 年 ...

  7. Jenkins创建任务进行构建项目配置

    总体构建项目的操作步骤 分为Generna(总的描述).源码管理.构建触发器.构建环境.构建.构建后的操作 1.Dashboard-> new item > 新建一个任务,选择freest ...

  8. Scrcpy - 开源免费在电脑显示手机画面并控制手机的工具 (投屏/录屏/免Root)

    教程:https://www.iplaysoft.com/scrcpy.html 官方地址:https://github.com/Genymobile/scrcpy

  9. FFmpeg开发笔记(三十二)利用RTMP协议构建电脑与手机的直播Demo

    不管是传统互联网还是移动互联网,实时数据传输都是刚需,比如以QQ.微信为代表的即时通信工具,能够实时传输文本和图片.其中一对一的图文通信叫做私聊,多对多的图文通信叫做群聊. 除了常见的图文即时通信,还 ...

  10. Linux系统基本介绍

    一.Linux系统基本介绍 [1].Linux操作系统特点 多任务的操作系统 多任务操作系统 严格区分大小写 Linux一切皆文件 所有文件的起点都是"/"根目录 对于Linux操 ...