SpringBoot?整合mapstruct的實(shí)現(xiàn)步驟
前言
在項(xiàng)目中,如果我們要遵循分層領(lǐng)域模型規(guī)約: 話,肯定避免不了在DTO、VO、BO、AO、VO、Query等實(shí)體的轉(zhuǎn)換,我們通常有幾種做法:
手動(dòng)一個(gè)個(gè)字段的賦值通過(guò)反序列化的手段,必須先轉(zhuǎn)成JSON字符串,再轉(zhuǎn)回來(lái)使用Spring的BeanUtils,提供的克隆方法
上面三種方式我們應(yīng)該都使用過(guò),但是我們今天介紹的主角是mapstruct,我們接下來(lái)見(jiàn)到介紹下它,以及為什么選擇它。
什么是DTO、VO、BO、AO、VO、Query
這里是摘錄自《阿里巴巴Java開(kāi)發(fā)規(guī)范》
- DO(Data Object):此對(duì)象與數(shù)據(jù)庫(kù)表結(jié)構(gòu)一一對(duì)應(yīng),通過(guò) DAO 層向上傳輸數(shù)據(jù)源對(duì)象。?
- DTO(Data Transfer Object):數(shù)據(jù)傳輸對(duì)象,Service 或 Manager 向外傳輸?shù)膶?duì)象。
- BO(Business Object):業(yè)務(wù)對(duì)象,由 Service 層輸出的封裝業(yè)務(wù)邏輯的對(duì)象。
- AO(ApplicationObject):應(yīng)用對(duì)象,在Web層與Service層之間抽象的復(fù)用對(duì)象模型, 極為貼近展示層,復(fù)用度不高。
- VO(View Object):顯示層對(duì)象,通常是 Web 向模板渲染引擎層傳輸?shù)膶?duì)象。
Query:數(shù)據(jù)查詢對(duì)象,各層接收上層的查詢請(qǐng)求。注意超過(guò) 2 個(gè)參數(shù)的查詢封裝,禁止
使用 Map 類來(lái)傳輸。
mapstruct 使用來(lái)干什么的?
通俗的來(lái)說(shuō),mapstruct就是用來(lái)做對(duì)象復(fù)制的
mapstruct 相對(duì)于BeanUtils的優(yōu)勢(shì)在哪?
- 支持復(fù)雜屬性賦值
- 效率高,在編譯時(shí)直接給你生成代碼,相當(dāng)與幫你手動(dòng)去一個(gè)個(gè)賦值
- 支持不同字段間的賦值,通過(guò)注解實(shí)現(xiàn)
編碼
引入依賴
項(xiàng)目中除了引用mapstruct本身的依賴 ,還引入了神器lombok,不用寫get set,其實(shí)這里也引了一個(gè)坑進(jìn)來(lái),相信同學(xué)應(yīng)該也碰到過(guò):
當(dāng)lombok和mapstruct一起用的時(shí)候,會(huì)導(dǎo)致mapstruct失效?
后面會(huì)幫助大家解決這個(gè)問(wèn)題。
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.6.0</version> <relativePath/> <!-- lookup parent from repository --> </parent> <groupId>com.ams</groupId> <artifactId>springboot-mapstruct</artifactId> <version>0.0.1-SNAPSHOT</version> <name>springboot-mapstruct</name> <description>Demo project for Spring Boot</description> <properties> <java.version>1.8</java.version> <org.mapstruct>1.4.1.Final</org.mapstruct> </properties> <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency> <!-- mapStruct 對(duì)象轉(zhuǎn)換 --> <dependency> <groupId>org.mapstruct</groupId> <artifactId>mapstruct</artifactId> <version>${org.mapstruct}</version> </dependency> <dependency> <groupId>org.mapstruct</groupId> <artifactId>mapstruct-processor</artifactId> <version>${org.mapstruct}</version> </dependency> <!-- 不是必備 只是為了懶,不用寫get set方法--> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <optional>true</optional> </dependency> <!-- <dependency>--> <!-- <groupId>org.projectlombok</groupId>--> <!-- <artifactId>lombok-mapstruct-binding</artifactId>--> <!-- <version>0.2.0</version>--> <!-- </dependency>--> </dependencies> <build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <configuration> <excludes> <exclude> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> </exclude> </excludes> </configuration> </plugin> </plugins> </build> </project>
創(chuàng)建 DTO、VO
StudentDto
package com.ams.springbootmapstruct.dto; import lombok.Data; /** * Created with IntelliJ IDEA. * * @author: AI碼師 * @date: 2021/11/27 * @description: * @modifiedBy: * @version: 1.0 */ @Data public class StudentDto { private String userName; private String userId; private String address; private String school; private int age; private String email; }
StudenVo
package com.ams.springbootmapstruct.vo; import lombok.Builder; import lombok.Data; /** * Created with IntelliJ IDEA. * * @author: AI碼師 * @date: 2021/11/27 * @description: * @modifiedBy: * @version: 1.0 */ @Data @Builder public class StudentVo { private String userName; private String userId; private String address; private String school; private int age; private String emailAddress; }
創(chuàng)建mapstruct轉(zhuǎn)換器
package com.ams.springbootmapstruct.mapper; import com.ams.springbootmapstruct.dto.StudentDto; import com.ams.springbootmapstruct.vo.StudentVo; import org.mapstruct.Mapper; @Mapper(componentModel = "spring") public interface MainMapper { StudentDto studentVo2Dto(StudentVo vo); }
編寫測(cè)試用例
package com.ams.springbootmapstruct; import com.ams.springbootmapstruct.dto.StudentDto; import com.ams.springbootmapstruct.mapper.MainMapper; import com.ams.springbootmapstruct.vo.StudentVo; import org.junit.jupiter.api.Test; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; @SpringBootTest class SpringbootMapstructApplicationTests { @Autowired private MainMapper mainMapper; @Test void testSimpleMap() { StudentVo studentVo = StudentVo.builder() .school("清華大學(xué)") .userId("ams") .userName("AI碼師") .age(27) .address("合肥") .build(); StudentDto studentDto = mainMapper.studentVo2Dto(studentVo); System.out.println(studentDto); } }
運(yùn)行測(cè)試用例
運(yùn)行test之后,發(fā)現(xiàn)輸出內(nèi)容是空的
這是怎么回事呢,我們看下MainMapper生成的代碼是什么樣的?
看到生成的代碼里面只是new了一個(gè)新的對(duì)象,并沒(méi)有做賦值操作。
這是為什么呢?
答案:由于mapstruct和lombok都會(huì)在編譯期為項(xiàng)目生成代碼,兩個(gè)如果一起用的話,就有可能導(dǎo)致mapstruct失效;我猜測(cè)有可能我們借助lombok生成 get set方法的原因,有可能mapstruct生成代碼之前,lombok還沒(méi)有生成get set方法,所以mapstruct也就調(diào)用不了get set 進(jìn)行賦值了。
怎么解決mapstruct 失效呢?
其實(shí)我們只需要引入一個(gè)依賴就可以了
<dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok-mapstruct-binding</artifactId> <version>0.2.0</version> </dependency>
重新運(yùn)行下,看是不是解決了
再看下生成的代碼,發(fā)現(xiàn)它已經(jīng)調(diào)用set方法賦值了
mapstruct常規(guī)操作
不同字段映射
如果兩個(gè)實(shí)體中 有幾個(gè)字段命名不一致,可以使用@Mapping 解決
現(xiàn)在studenVo和studenDto 有email 和emailAddress 字段不一致,可以使用如下方式解決
package com.ams.springbootmapstruct.mapper; import com.ams.springbootmapstruct.dto.StudentDto; import com.ams.springbootmapstruct.vo.StudentVo; import org.mapstruct.Mapper; import org.mapstruct.Mapping; @Mapper(componentModel = "spring") public interface MainMapper { @Mapping(source = "emailAddress", target = "email") StudentDto studentVo2Dto(StudentVo vo); }
LIST轉(zhuǎn)換
package com.ams.springbootmapstruct.mapper; import com.ams.springbootmapstruct.dto.StudentDto; import com.ams.springbootmapstruct.vo.StudentVo; import org.mapstruct.Mapper; import org.mapstruct.Mapping; import java.util.List; @Mapper(componentModel = "spring") public interface MainMapper { @Mapping(source = "emailAddress", target = "email") StudentDto studentVo2Dto(StudentVo vo); List<StudentDto> studentListVo2Dto(List<StudentVo> vo); }
總結(jié)
本文整理了SpringBoot集成mapstruct的基本過(guò)程,解決了mapstruct和lombok一起使用,導(dǎo)致mapstruct失效的bug,另外也介紹了mapstruct的基本使用方法,后續(xù)會(huì)出更多集成指南,敬請(qǐng)期待!
代碼已經(jīng)上傳到碼云:https://gitee.com/lezaiclub/springboot-hyper-integration.git,歡迎白嫖
到此這篇關(guān)于SpringBoot 整合mapstruct的文章就介紹到這了,更多相關(guān)SpringBoot 整合mapstruct內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java使用elasticsearch基礎(chǔ)API使用案例講解
這篇文章主要介紹了Java使用elasticsearch基礎(chǔ)API使用案例講解,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08Java框架Struts2實(shí)現(xiàn)圖片上傳功能
這篇文章主要為大家詳細(xì)介紹了Java框架Struts2實(shí)現(xiàn)圖片上傳功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-08-08springboot?jpa?實(shí)現(xiàn)返回結(jié)果自定義查詢
這篇文章主要介紹了springboot?jpa?實(shí)現(xiàn)返回結(jié)果自定義查詢方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02Spring BeanFactory和FactoryBean有哪些區(qū)別
這篇文章主要介紹了Spring BeanFactory 與 FactoryBean 的區(qū)別詳情,BeanFactory 和 FactoryBean 的區(qū)別卻是一個(gè)很重要的知識(shí)點(diǎn),在本文中將結(jié)合源碼進(jìn)行分析講解,需要的小伙伴可以參考一下2023-02-02詳解使用spring boot admin監(jiān)控spring cloud應(yīng)用程序
這篇文章主要介紹了詳解使用spring boot admin監(jiān)控spring cloud應(yīng)用程序,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-05-05java實(shí)現(xiàn)簡(jiǎn)單的搜索引擎
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)簡(jiǎn)單的搜索引擎的相關(guān)資料,需要的朋友可以參考下2016-02-02springboot集成普羅米修斯(Prometheus)的方法
這篇文章主要介紹了springboot集成普羅米修斯(Prometheus)的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08深入解析Java的Spring框架中的混合事務(wù)與bean的區(qū)分
這篇文章主要介紹了Java的Spring框架中的混合事務(wù)與bean的區(qū)分,Spring是Java的SSH三大web開(kāi)發(fā)框架之一,需要的朋友可以參考下2016-01-01