干掉 BeanUtils!試試這款 Bean 自動映射工具,真心強大?。?/h1>
開發(fā)背景
你有沒有遇到過這樣的開發(fā)場景?
服務(wù)通過接口對外提供數(shù)據(jù),或者服務(wù)之間進(jìn)行數(shù)據(jù)交互,首先查詢數(shù)據(jù)庫并映射成數(shù)據(jù)對象(XxxDO)。
正常情況下,接口是不允許直接以數(shù)據(jù)庫數(shù)據(jù)對象 XxxDO 形式對外提供數(shù)據(jù)的,而是要再封裝成數(shù)據(jù)傳輸對象(XxxDTO)提供出去。
為什么不能直接提供 DO?
1)根據(jù)單一設(shè)計原則,DO 只能對應(yīng)數(shù)據(jù)實體對象,不能承擔(dān)其他職責(zé);
2)DO 可能包含表所有字段數(shù)據(jù),不符合接口的參數(shù)定義,數(shù)據(jù)如果過大會影響傳輸速度,也不符合數(shù)據(jù)安全原則;
3)根據(jù)《阿里 Java 開發(fā)手冊》分層領(lǐng)域模型規(guī)約,不能一個對象走天下,需要定義成 POJO/DO/BO/DTO/VO/Query 等數(shù)據(jù)對象,完整的定義可以參考阿里開發(fā)手冊,關(guān)注公眾號:Java技術(shù)棧,在后臺回復(fù):手冊,可以獲取最新高清完整版。
傳統(tǒng) DO -> DTO 做法
XxxDTO 可能包含 XxxDO 大部分?jǐn)?shù)據(jù),或者組合其他 DO 的部分?jǐn)?shù)據(jù),傳統(tǒng)的做法有以下幾種:
- get/ set
- 構(gòu)造器
- BeanUtils 工具類
- Builder 模式
我相信大部分人的做法都是這樣的,雖然很直接,但是普遍真的很 Low,耦合性又強,還經(jīng)常丟參數(shù),或者搞錯參數(shù)值,在這個開發(fā)場景,我個人覺得這些都不是最佳的方式。
這種開發(fā)場景又實在是太常見了,那有沒有一種 Java bean 自動映射工具?
沒錯——正是 MapStruct!!
MapStruct 簡介
官網(wǎng)地址:
https://mapstruct.org/
開源地址:
https://github.com/mapstruct/mapstruct
Java bean mappings, the easy way!
以簡單的方式進(jìn)行 Java bean 映射。
MapStruct 是一個代碼生成器,它和 Spring Boot、Maven 一樣也是基于約定優(yōu)于配置的理念,極大地簡化了 Java bean 之間數(shù)據(jù)映射的實現(xiàn)。
MapStruct 的優(yōu)勢:
1、MapStruct 使用簡單的方法調(diào)用生成映射代碼,因此***速度非常快***;
2、類型安全,避免出錯,只能映射相互映射的對象和屬性,因此不會錯誤將用戶實體錯誤地映射到訂單 DTO;
3、只需要 JDK 1.8+,不用其他任何依賴,自包含所有代碼;
4、易于調(diào)試;
5、易于理解;
支持的方式:
MapStruct 支持命令行編譯,如:純 javac 命令、Maven、Gradle、Ant 等等,也支持 Eclipse、IntelliJ IDEA 等 IDEs。
MapStruct 實戰(zhàn)
本文棧長基于 IntelliJ IDEA、Spring Boot、Maven 進(jìn)行演示。
基本準(zhǔn)備
新增兩個數(shù)據(jù)庫 DO 類:
一個用戶主類,一個用戶擴展類。
- /**
- * 微信公眾號:Java技術(shù)棧
- * @author 棧長
- */
- @Data
- public class UserDO {
- private String name;
- private int sex;
- private int age;
- private Date birthday;
- private String phone;
- private boolean married;
- private Date regDate;
- private Date loginDate;
- private String memo;
- private UserExtDO userExtDO;
- }
- /**
- * 微信公眾號:Java技術(shù)棧
- * @author 棧長
- */
- @Data
- public class UserExtDO {
- private String regSource;
- private String favorite;
- private String school;
- private int kids;
- private String memo;
- }
新增一個數(shù)據(jù)傳輸 DTO 類:
用戶展示類,包含用戶主類、用戶擴展類的部分?jǐn)?shù)據(jù)。
- /**
- * 微信公眾號:Java技術(shù)棧
- * @author 棧長
- */
- @Data
- public class UserShowDTO {
- private String name;
- private int sex;
- private boolean married;
- private String birthday;
- private String regDate;
- private String registerSource;
- private String favorite;
- private String memo;
- }
開始實戰(zhàn)
重點來了,不要 get/set,不要 BeanUtils,怎么把兩個用戶對象的數(shù)據(jù)封裝到 DTO 對象?
Spring Boot 基礎(chǔ)這篇就不介紹了,系列基礎(chǔ)教程和示例源碼可以看這里:https://github.com/javastacks/spring-boot-best-practice
引入 MapStruct 依賴:
- <dependencies>
- <dependency>
- <groupId>org.mapstruct</groupId>
- <artifactId>mapstruct</artifactId>
- <version>${org.mapstruct.version}</version>
- </dependency>
- </dependencies>
Maven 插件相關(guān)配置:
MapStruct 和 Lombok 結(jié)合使用會有版本沖突問題,注意以下配置。
- <build>
- <plugins>
- <plugin>
- <groupId>org.apache.maven.plugins</groupId>
- <artifactId>maven-compiler-plugin</artifactId>
- <version>3.8.1</version>
- <configuration>
- <source>1.8</source>
- <target>1.8</target>
- <annotationProcessorPaths>
- <path>
- <groupId>org.mapstruct</groupId>
- <artifactId>mapstruct-processor</artifactId>
- <version>${org.mapstruct.version}</version>
- </path>
- <!-- 使用 Lombok 需要添加 -->
- <path>
- <groupId>org.projectlombok</groupId>
- <artifactId>lombok</artifactId>
- <version>${org.projectlombok.version}</version>
- </path>
- <!-- Lombok 1.18.16 及以上需要添加,不然報錯 -->
- <path>
- <groupId>org.projectlombok</groupId>
- <artifactId>lombok-mapstruct-binding</artifactId>
- <version>${lombok-mapstruct-binding.version}</version>
- </path>
- </annotationProcessorPaths>
- </configuration>
- </plugin>
- </plugins>
- </build>
添加 MapStruct 映射:
- /**
- * 微信公眾號:Java技術(shù)棧
- * @author 棧長
- */
- @Mapper
- public interface UserStruct {
- UserStruct INSTANCE = Mappers.getMapper(UserStruct.class);
- @Mappings({
- @Mapping(source = "birthday", target = "birthday", dateFormat = "yyyy-MM-dd")
- @Mapping(target = "regDate", expression = "java(org.apache.commons.lang3.time.DateFormatUtils.format(userDO.getRegDate(),\"yyyy-MM-dd HH:mm:ss\"))")
- @Mapping(source = "userExtDO.regSource", target = "registerSource")
- @Mapping(source = "userExtDO.favorite", target = "favorite")
- @Mapping(target = "memo", ignore = true)
- })
- UserShowDTO toUserShowDTO(UserDO userDO);
- List<UserShowDTO> toUserShowDTOs(List<UserDO> userDOs);
- }
重點說明:
1)添加一個 interface 接口,使用 MapStruct 的 @Mapper 注解修飾,這里取名 XxxStruct,是為了不和 MyBatis 的 Mapper 混淆;
2)使用 Mappers 添加一個 INSTANCE 實例,也可以使用 Spring 注入,后面會講到;
3)添加兩個映射方法,返回單個對象、對象列表;
4)使用 @Mappings + @Mapping 組合映射,如果兩個字段名相同可以不用寫,可以指定映射的日期格式、數(shù)字格式、表達(dá)式等,ignore 表示忽略該字段映射;
5)List 方法的映射會調(diào)用單個方法映射,不用單獨映射,后面看源碼就知道了;
另外,Java 8+ 以上版本不需要 @Mappings 注解,直接使用 @Mapping 注解就行了:
Java 8 修改之后:
- /**
- * 微信公眾號:Java技術(shù)棧
- * @author 棧長
- */
- @Mapper
- public interface UserStruct {
- UserStruct INSTANCE = Mappers.getMapper(UserStruct.class);
- @Mapping(source = "birthday", target = "birthday", dateFormat = "yyyy-MM-dd")
- @Mapping(target = "regDate", expression = "java(org.apache.commons.lang3.time.DateFormatUtils.format(userDO.getRegDate(),\"yyyy-MM-dd HH:mm:ss\"))")
- @Mapping(source = "userExtDO.regSource", target = "registerSource")
- @Mapping(source = "userExtDO.favorite", target = "favorite")
- @Mapping(target = "memo", ignore = true)
- UserShowDTO toUserShowDTO(UserDO userDO);
- List<UserShowDTO> toUserShowDTOs(List<UserDO> userDOs);
- }
測試一下:
- /**
- * 微信公眾號:Java技術(shù)棧
- * @author 棧長
- */
- public class UserStructTest {
- @Test
- public void test1() {
- UserExtDO userExtDO = new UserExtDO();
- userExtDO.setRegSource("公眾號:Java技術(shù)棧");
- userExtDO.setFavorite("寫代碼");
- userExtDO.setSchool("社會大學(xué)");
- UserDO userDO = new UserDO();
- userDO.setName("棧長");
- userDO.setSex(1);
- userDO.setAge(18);
- userDO.setBirthday(new Date());
- userDO.setPhone("18888888888");
- userDO.setMarried(true);
- userDO.setRegDate(new Date());
- userDO.setMemo("666");
- userDO.setUserExtDO(userExtDO);
- UserShowDTO userShowDTO = UserStruct.INSTANCE.toUserShowDTO(userDO);
- System.out.println("=====單個對象映射=====");
- System.out.println(userShowDTO);
- List<UserDO> userDOs = new ArrayList<>();
- UserDO userDO2 = new UserDO();
- BeanUtils.copyProperties(userDO, userDO2);
- userDO2.setName("棧長2");
- userDOs.add(userDO);
- userDOs.add(userDO2);
- List<UserShowDTO> userShowDTOs = UserStruct.INSTANCE.toUserShowDTOs(userDOs);
- System.out.println("=====對象列表映射=====");
- userShowDTOs.forEach(System.out::println);
- }
- }
輸出結(jié)果:
來看結(jié)果,數(shù)據(jù)轉(zhuǎn)換結(jié)果成功。
什么原理?
如上我們知道,通過一個注解修飾接口就可以搞定了,是什么原理呢?
來看編譯后的目錄:
原理就是在編譯期間生成了一個該接口的實現(xiàn)類。
打開看下其源碼:
- public class UserStructImpl implements UserStruct {
- public UserStructImpl() {
- }
- public UserShowDTO toUserShowDTO(UserDO userDO) {
- if (userDO == null) {
- return null;
- } else {
- UserShowDTO userShowDTO = new UserShowDTO();
- if (userDO.getBirthday() != null) {
- userShowDTO.setBirthday((new SimpleDateFormat("yyyy-MM-dd")).format(userDO.getBirthday()));
- }
- userShowDTO.setRegisterSource(this.userDOUserExtDORegSource(userDO));
- userShowDTO.setFavorite(this.userDOUserExtDOFavorite(userDO));
- userShowDTO.setName(userDO.getName());
- userShowDTO.setSex(userDO.getSex());
- userShowDTO.setMarried(userDO.isMarried());
- userShowDTO.setRegDate(DateFormatUtils.format(userDO.getRegDate(), "yyyy-MM-dd HH:mm:ss"));
- return userShowDTO;
- }
- }
- public List<UserShowDTO> toUserShowDTOs(List<UserDO> userDOs) {
- if (userDOs == null) {
- return null;
- } else {
- List<UserShowDTO> list = new ArrayList(userDOs.size());
- Iterator var3 = userDOs.iterator();
- while(var3.hasNext()) {
- UserDO userDO = (UserDO)var3.next();
- list.add(this.toUserShowDTO(userDO));
- }
- return list;
- }
- }
- private String userDOUserExtDORegSource(UserDO userDO) {
- if (userDO == null) {
- return null;
- } else {
- UserExtDO userExtDO = userDO.getUserExtDO();
- if (userExtDO == null) {
- return null;
- } else {
- String regSource = userExtDO.getRegSource();
- return regSource == null ? null : regSource;
- }
- }
- }
- private String userDOUserExtDOFavorite(UserDO userDO) {
- if (userDO == null) {
- return null;
- } else {
- UserExtDO userExtDO = userDO.getUserExtDO();
- if (userExtDO == null) {
- return null;
- } else {
- String favorite = userExtDO.getFavorite();
- return favorite == null ? null : favorite;
- }
- }
- }
- }
其實實現(xiàn)類就是調(diào)用了對象的 get/set 等其他常規(guī)操作,而 List 就是循環(huán)調(diào)用的該對象的單個映射方法,這下就清楚了吧!
Spring 注入法
上面的示例創(chuàng)建了一個 UserStruct 實例:
- UserStruct INSTANCE = Mappers.getMapper(UserStruct.class);
如 @Mapper 注解源碼所示:
參數(shù) componentModel 默認(rèn)值是 default,也就是手動創(chuàng)建實例,也可以通過 Spring 注入。
Spring 修改版如下:
干掉了 INSTANCE,@Mapper 注解加入了 componentModel = "spring" 值。
- /**
- * 微信公眾號:Java技術(shù)棧
- * @author 棧長
- */
- @Mapper(componentModel = "spring")
- public interface UserSpringStruct {
- @Mapping(source = "birthday", target = "birthday", dateFormat = "yyyy-MM-dd")
- @Mapping(target = "regDate", expression = "java(org.apache.commons.lang3.time.DateFormatUtils.format(userDO.getRegDate(),\"yyyy-MM-dd HH:mm:ss\"))")
- @Mapping(source = "userExtDO.regSource", target = "registerSource")
- @Mapping(source = "userExtDO.favorite", target = "favorite")
- @Mapping(target = "memo", ignore = true)
- UserShowDTO toUserShowDTO(UserDO userDO);
- List<UserShowDTO> toUserShowDTOs(List<UserDO> userDOS);
- }
測試一下:
本文用到了 Spring Boot,所以這里就要用到 Spring Boot 的單元測試方法。Spring Boot 單元測試不懂的可以關(guān)注公眾號:Java技術(shù)棧,在后臺回復(fù):boot,系列教程都整理好了。
- /**
- * 微信公眾號:Java技術(shù)棧
- * @author 棧長
- */
- @RunWith(SpringRunner.class)
- @SpringBootTest
- public class UserSpringStructTest {
- @Autowired
- private UserSpringStruct userSpringStruct;
- @Test
- public void test1() {
- UserExtDO userExtDO = new UserExtDO();
- userExtDO.setRegSource("公眾號:Java技術(shù)棧");
- userExtDO.setFavorite("寫代碼");
- userExtDO.setSchool("社會大學(xué)");
- UserDO userDO = new UserDO();
- userDO.setName("棧長Spring");
- userDO.setSex(1);
- userDO.setAge(18);
- userDO.setBirthday(new Date());
- userDO.setPhone("18888888888");
- userDO.setMarried(true);
- userDO.setRegDate(new Date());
- userDO.setMemo("666");
- userDO.setUserExtDO(userExtDO);
- UserShowDTO userShowDTO = userSpringStruct.toUserShowDTO(userDO);
- System.out.println("=====單個對象映射=====");
- System.out.println(userShowDTO);
- List<UserDO> userDOs = new ArrayList<>();
- UserDO userDO2 = new UserDO();
- BeanUtils.copyProperties(userDO, userDO2);
- userDO2.setName("棧長Spring2");
- userDOs.add(userDO);
- userDOs.add(userDO2);
- List<UserShowDTO> userShowDTOs = userSpringStruct.toUserShowDTOs(userDOs);
- System.out.println("=====對象列表映射=====");
- userShowDTOs.forEach(System.out::println);
- }
- }
如上所示,直接使用 @Autowired 注入就行,使用更方便。
輸出結(jié)果:
沒毛病,穩(wěn)如狗。
總結(jié)
本文棧長只是介紹了 MapStruct 的簡單用法,使用 MapStruct 可以使代碼更優(yōu)雅,還能避免出錯,其實還有很多復(fù)雜的、個性化用法,一篇難以寫完,棧長后面有時間會整理出來,陸續(xù)給大家分享。
感興趣的也可以參考官方文檔:
https://mapstruct.org/documentation/reference-guide/
本文實戰(zhàn)源代碼完整版已經(jīng)上傳:
https://github.com/javastacks/spring-boot-best-practice
歡迎 Star 學(xué)習(xí),后面 Spring Boot 示例都會在這上面提供!
本文轉(zhuǎn)載自微信公眾號「Java技術(shù)?!?,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請聯(lián)系Java技術(shù)棧公眾號。