Java
首先来了解一下DTO,DTO简单的理解就是做数据传输对象的,类似于VO,但是VO用于传输到前端。(~~)

1、MapStruct是用来做什么的?

现在有这么个场景,从数据库查询出来了一个user对象(包含id,用户名,密码,手机号,邮箱,角色这些字段)和一个对应的角色对象role(包含id,角色名,角色描述这些字段),现在在controller需要用到user对象的id,用户名,和角色对象的角色名三个属性。
一种方式是直接把两个对象传递到controller层,但是这样会多出很多没用的属性。更通用的方式是需要用到的属性封装成一个类(DTO),通过传输这个类的实例来完成数据传输。
User.java

  1. @AllArgsConstructor
  2. @Data
  3. public class User {
  4. private Long id;
  5. private String username;
  6. private String password;
  7. private String phoneNum;
  8. private String email;
  9. private Role role;
  10. }

Role.java

  1. @AllArgsConstructor
  2. @Data
  3. public class Role {
  4. private Long id;
  5. private String roleName;
  6. private String description;
  7. }

UserRoleDto.java,这个类就是封装的类

  1. @Data
  2. public class UserRoleDto {
  3. /**
  4. * 用户id
  5. */
  6. private Long userId;
  7. /**
  8. * 用户名
  9. */
  10. private String name;
  11. /**
  12. * 角色名
  13. */
  14. private String roleName;
  15. }

测试类,模拟将user对象转换成UserRoleDto对象

  1. public class MainTest {
  2. User user = null;
  3. /**
  4. * 模拟从数据库中查出user对象
  5. */
  6. @Before
  7. public void before() {
  8. Role role = new Role(2L, "administrator", "超级管理员");
  9. user = new User(1L, "zhangsan", "12345", "17677778888", "123@qq.com", role);
  10. }
  11. /**
  12. * 模拟把user对象转换成UserRoleDto对象
  13. */
  14. @Test
  15. public void test1() {
  16. UserRoleDto userRoleDto = new UserRoleDto();
  17. userRoleDto.setUserId(user.getId());
  18. userRoleDto.setName(user.getUsername());
  19. userRoleDto.setRoleName(user.getRole().getRoleName());
  20. System.out.println(userRoleDto);
  21. }
  22. }

从上面代码可以看出,通过getter、setter的方式把一个对象属性值复制到另一个对象中去还是很麻烦的,尤其是当属性过多的时候。而MapStruct就是用于解决这种问题的。

2、使用MapStruct解决上述问题

这里沿用User.java、Role.java、UserRoleDto.java。
新建一个UserRoleMapper.java,这个来用来定义User.java、Role.java和UserRoleDto.java之间属性对应规则:
UserRoleMapper.java

  1. import org.mapstruct.Mapper;
  2. import org.mapstruct.Mapping;
  3. import org.mapstruct.Mappings;
  4. import org.mapstruct.factory.Mappers;
  5. /**
  6. * @Mapper 定义这是一个MapStruct对象属性转换接口,在这个类里面规定转换规则
  7. * 在项目构建时,会自动生成改接口的实现类,这个实现类将实现对象属性值复制
  8. */
  9. @Mapper
  10. public interface UserRoleMapper {
  11. /**
  12. * 获取该类自动生成的实现类的实例
  13. * 接口中的属性都是 public static final 的 方法都是public abstract的
  14. */
  15. UserRoleMapper INSTANCES = Mappers.getMapper(UserRoleMapper.class);
  16. /**
  17. * 这个方法就是用于实现对象属性复制的方法
  18. *
  19. * @Mapping 用来定义属性复制规则 source 指定源对象属性 target指定目标对象属性
  20. *
  21. * @param user 这个参数就是源对象,也就是需要被复制的对象
  22. * @return 返回的是目标对象,就是最终的结果对象
  23. */
  24. @Mappings({
  25. @Mapping(source = "id", target = "userId"),
  26. @Mapping(source = "username", target = "name"),
  27. @Mapping(source = "role.roleName", target = "roleName")
  28. })
  29. UserRoleDto toUserRoleDto(User user);
  30. }

通过上面的例子可以看出,使用MapStruct方便许多。

3、添加默认方法

添加默认方法是为了这个类(接口)不只是为了做数据转换用的,也可以做一些其他的事。

  1. import org.mapstruct.Mapper;
  2. import org.mapstruct.Mapping;
  3. import org.mapstruct.Mappings;
  4. import org.mapstruct.factory.Mappers;
  5. /**
  6. * @Mapper 定义这是一个MapStruct对象属性转换接口,在这个类里面规定转换规则
  7. * 在项目构建时,会自动生成改接口的实现类,这个实现类将实现对象属性值复制
  8. */
  9. @Mapper
  10. public interface UserRoleMapper {
  11. /**
  12. * 获取该类自动生成的实现类的实例
  13. * 接口中的属性都是 public static final 的 方法都是public abstract的
  14. */
  15. UserRoleMapper INSTANCES = Mappers.getMapper(UserRoleMapper.class);
  16. /**
  17. * 这个方法就是用于实现对象属性复制的方法
  18. *
  19. * @Mapping 用来定义属性复制规则 source 指定源对象属性 target指定目标对象属性
  20. *
  21. * @param user 这个参数就是源对象,也就是需要被复制的对象
  22. * @return 返回的是目标对象,就是最终的结果对象
  23. */
  24. @Mappings({
  25. @Mapping(source = "id", target = "userId"),
  26. @Mapping(source = "username", target = "name"),
  27. @Mapping(source = "role.roleName", target = "roleName")
  28. })
  29. UserRoleDto toUserRoleDto(User user);
  30. /**
  31. * 提供默认方法,方法自己定义,这个方法是我随便写的,不是要按照这个格式来的
  32. * @return
  33. */
  34. default UserRoleDto defaultConvert() {
  35. UserRoleDto userRoleDto = new UserRoleDto();
  36. userRoleDto.setUserId(0L);
  37. userRoleDto.setName("None");
  38. userRoleDto.setRoleName("None");
  39. return userRoleDto;
  40. }
  41. }

测试代码:

  1. @Test
  2. public void test3() {
  3. UserRoleMapper userRoleMapperInstances = UserRoleMapper.INSTANCES;
  4. UserRoleDto userRoleDto = userRoleMapperInstances.defaultConvert();
  5. System.out.println(userRoleDto);
  6. }

4、可以使用abstract class来代替接口

mapper可以用接口来实现,也可以完全由抽象来完全代替

  1. import org.mapstruct.Mapper;
  2. import org.mapstruct.Mapping;
  3. import org.mapstruct.Mappings;
  4. import org.mapstruct.factory.Mappers;
  5. /**
  6. * @Mapper 定义这是一个MapStruct对象属性转换接口,在这个类里面规定转换规则
  7. * 在项目构建时,会自动生成改接口的实现类,这个实现类将实现对象属性值复制
  8. */
  9. @Mapper
  10. public abstract class UserRoleMapper {
  11. /**
  12. * 获取该类自动生成的实现类的实例
  13. * 接口中的属性都是 public static final 的 方法都是public abstract的
  14. */
  15. public static final UserRoleMapper INSTANCES = Mappers.getMapper(UserRoleMapper.class);
  16. /**
  17. * 这个方法就是用于实现对象属性复制的方法
  18. *
  19. * @Mapping 用来定义属性复制规则 source 指定源对象属性 target指定目标对象属性
  20. *
  21. * @param user 这个参数就是源对象,也就是需要被复制的对象
  22. * @return 返回的是目标对象,就是最终的结果对象
  23. */
  24. @Mappings({
  25. @Mapping(source = "id", target = "userId"),
  26. @Mapping(source = "username", target = "name"),
  27. @Mapping(source = "role.roleName", target = "roleName")
  28. })
  29. public abstract UserRoleDto toUserRoleDto(User user);
  30. /**
  31. * 提供默认方法,方法自己定义,这个方法是我随便写的,不是要按照这个格式来的
  32. * @return
  33. */
  34. UserRoleDto defaultConvert() {
  35. UserRoleDto userRoleDto = new UserRoleDto();
  36. userRoleDto.setUserId(0L);
  37. userRoleDto.setName("None");
  38. userRoleDto.setRoleName("None");
  39. return userRoleDto;
  40. }
  41. }

5、可以使用多个参数

可以绑定多个对象的属性值到目标对象中:

  1. package com.mapstruct.demo;
  2. import org.mapstruct.Mapper;
  3. import org.mapstruct.Mapping;
  4. import org.mapstruct.Mappings;
  5. import org.mapstruct.factory.Mappers;
  6. /**
  7. * @Mapper 定义这是一个MapStruct对象属性转换接口,在这个类里面规定转换规则
  8. * 在项目构建时,会自动生成改接口的实现类,这个实现类将实现对象属性值复制
  9. */
  10. @Mapper
  11. public interface UserRoleMapper {
  12. /**
  13. * 获取该类自动生成的实现类的实例
  14. * 接口中的属性都是 public static final 的 方法都是public abstract的
  15. */
  16. UserRoleMapper INSTANCES = Mappers.getMapper(UserRoleMapper.class);
  17. /**
  18. * 这个方法就是用于实现对象属性复制的方法
  19. *
  20. * @Mapping 用来定义属性复制规则 source 指定源对象属性 target指定目标对象属性
  21. *
  22. * @param user 这个参数就是源对象,也就是需要被复制的对象
  23. * @return 返回的是目标对象,就是最终的结果对象
  24. */
  25. @Mappings({
  26. @Mapping(source = "id", target = "userId"),
  27. @Mapping(source = "username", target = "name"),
  28. @Mapping(source = "role.roleName", target = "roleName")
  29. })
  30. UserRoleDto toUserRoleDto(User user);
  31. /**
  32. * 多个参数中的值绑定
  33. * @param user 源1
  34. * @param role 源2
  35. * @return 从源1、2中提取出的结果
  36. */
  37. @Mappings({
  38. @Mapping(source = "user.id", target = "userId"), // 把user中的id绑定到目标对象的userId属性中
  39. @Mapping(source = "user.username", target = "name"), // 把user中的username绑定到目标对象的name属性中
  40. @Mapping(source = "role.roleName", target = "roleName") // 把role对象的roleName属性值绑定到目标对象的roleName中
  41. })
  42. UserRoleDto toUserRoleDto(User user, Role role);
  43. }

对比两个方法~

6、直接使用参数作为属性值

  1. package com.mapstruct.demo;
  2. import org.mapstruct.Mapper;
  3. import org.mapstruct.Mapping;
  4. import org.mapstruct.Mappings;
  5. import org.mapstruct.factory.Mappers;
  6. /**
  7. * @Mapper 定义这是一个MapStruct对象属性转换接口,在这个类里面规定转换规则
  8. * 在项目构建时,会自动生成改接口的实现类,这个实现类将实现对象属性值复制
  9. */
  10. @Mapper
  11. public interface UserRoleMapper {
  12. /**
  13. * 获取该类自动生成的实现类的实例
  14. * 接口中的属性都是 public static final 的 方法都是public abstract的
  15. */
  16. UserRoleMapper INSTANCES = Mappers.getMapper(UserRoleMapper.class);
  17. /**
  18. * 直接使用参数作为值
  19. * @param user
  20. * @param myRoleName
  21. * @return
  22. */
  23. @Mappings({
  24. @Mapping(source = "user.id", target = "userId"), // 把user中的id绑定到目标对象的userId属性中
  25. @Mapping(source = "user.username", target = "name"), // 把user中的username绑定到目标对象的name属性中
  26. @Mapping(source = "myRoleName", target = "roleName") // 把role对象的roleName属性值绑定到目标对象的roleName中
  27. })
  28. UserRoleDto useParameter(User user, String myRoleName);
  29. }

测试类:

  1. public class Test1 {
  2. Role role = null;
  3. User user = null;
  4. @Before
  5. public void before() {
  6. role = new Role(2L, "administrator", "超级管理员");
  7. user = new User(1L, "zhangsan", "12345", "17677778888", "123@qq.com", role);
  8. }
  9. @Test
  10. public void test1() {
  11. UserRoleMapper instances = UserRoleMapper.INSTANCES;
  12. UserRoleDto userRoleDto = instances.useParameter(user, "myUserRole");
  13. System.out.println(userRoleDto);
  14. }
  15. }

7、更新对象属性

在之前的例子中UserRoleDto useParameter(User user, String myRoleName);都是通过类似上面的方法来生成一个对象。而MapStruct提供了另外一种方式来更新一个对象中的属性。@MappingTarget

  1. public interface UserRoleMapper1 {
  2. UserRoleMapper1 INSTANCES = Mappers.getMapper(UserRoleMapper1.class);
  3. @Mappings({
  4. @Mapping(source = "userId", target = "id"),
  5. @Mapping(source = "name", target = "username"),
  6. @Mapping(source = "roleName", target = "role.roleName")
  7. })
  8. void updateDto(UserRoleDto userRoleDto, @MappingTarget User user);
  9. @Mappings({
  10. @Mapping(source = "id", target = "userId"),
  11. @Mapping(source = "username", target = "name"),
  12. @Mapping(source = "role.roleName", target = "roleName")
  13. })
  14. void update(User user, @MappingTarget UserRoleDto userRoleDto);
  15. }

通过@MappingTarget来指定目标类是谁(谁的属性需要被更新)。@Mapping还是用来定义属性对应规则。
以此为例说明:

  1. @Mappings({
  2. @Mapping(source = "id", target = "userId"),
  3. @Mapping(source = "username", target = "name"),
  4. @Mapping(source = "role.roleName", target = "roleName")
  5. })
  6. void update(User user, @MappingTarget UserRoleDto userRoleDto);

@MappingTarget标注的类UserRoleDto 为目标类,user类为源类,调用此方法,会把源类中的属性更新到目标类中。更新规则还是由@Mapping指定。

8、没有getter/setter也能赋值

对于没有getter/setter的属性也能实现赋值操作

  1. public class Customer {
  2. private Long id;
  3. private String name;
  4. //getters and setter omitted for brevity
  5. }
  6. public class CustomerDto {
  7. public Long id;
  8. public String customerName;
  9. }
  10. @Mapper
  11. public interface CustomerMapper {
  12. CustomerMapper INSTANCE = Mappers.getMapper( CustomerMapper.class );
  13. @Mapping(source = "customerName", target = "name")
  14. Customer toCustomer(CustomerDto customerDto);
  15. @InheritInverseConfiguration
  16. CustomerDto fromCustomer(Customer customer);
  17. }

@Mapping(source = "customerName", target = "name")不是用来指定属性映射的,如果两个对象的属性名相同是可以省略@Mapping的。
MapStruct生成的实现类:

  1. @Generated(
  2. value = "org.mapstruct.ap.MappingProcessor",
  3. date = "2019-02-14T15:41:21+0800",
  4. comments = "version: 1.3.0.Final, compiler: javac, environment: Java 1.8.0_181 (Oracle Corporation)"
  5. )
  6. public class CustomerMapperImpl implements CustomerMapper {
  7. @Override
  8. public Customer toCustomer(CustomerDto customerDto) {
  9. if ( customerDto == null ) {
  10. return null;
  11. }
  12. Customer customer = new Customer();
  13. customer.setName( customerDto.customerName );
  14. customer.setId( customerDto.id );
  15. return customer;
  16. }
  17. @Override
  18. public CustomerDto toCustomerDto(Customer customer) {
  19. if ( customer == null ) {
  20. return null;
  21. }
  22. CustomerDto customerDto = new CustomerDto();
  23. customerDto.customerName = customer.getName();
  24. customerDto.id = customer.getId();
  25. return customerDto;
  26. }
  27. }

@InheritInverseConfiguration在这里的作用就是实现customerDto.customerName = customer.getName();功能的。如果没有这个注解,toCustomerDto这个方法则不会有customerName 和name两个属性的对应关系的。

9、使用Spring依赖注入

  1. @Data
  2. @NoArgsConstructor
  3. @AllArgsConstructor
  4. public class Customer {
  5. private Long id;
  6. private String name;
  7. }
  8. @Data
  9. public class CustomerDto {
  10. private Long id;
  11. private String customerName;
  12. }
  13. // 这里主要是这个componentModel 属性,它的值就是当前要使用的依赖注入的环境
  14. @Mapper(componentModel = "spring")
  15. public interface CustomerMapper {
  16. @Mapping(source = "name", target = "customerName")
  17. CustomerDto toCustomerDto(Customer customer);
  18. }

@Mapper(componentModel = "spring"),表示把当前Mapper类纳入Spring容器。可以在其它类中直接注入了:

  1. @SpringBootApplication
  2. @RestController
  3. public class DemoMapstructApplication {
  4. // 注入Mapper
  5. @Autowired
  6. private CustomerMapper mapper;
  7. public static void main(String[] args) {
  8. SpringApplication.run(DemoMapstructApplication.class, args);
  9. }
  10. @GetMapping("/test")
  11. public String test() {
  12. Customer customer = new Customer(1L, "zhangsan");
  13. CustomerDto customerDto = mapper.toCustomerDto(customer);
  14. return customerDto.toString();
  15. }
  16. }

看一下由mapstruct自动生成的类文件,会发现标记了@Component注解。

  1. @Generated(
  2. value = "org.mapstruct.ap.MappingProcessor",
  3. date = "2019-02-14T15:54:17+0800",
  4. comments = "version: 1.3.0.Final, compiler: javac, environment: Java 1.8.0_181 (Oracle Corporation)"
  5. )
  6. @Component
  7. public class CustomerMapperImpl implements CustomerMapper {
  8. @Override
  9. public CustomerDto toCustomerDto(Customer customer) {
  10. if ( customer == null ) {
  11. return null;
  12. }
  13. CustomerDto customerDto = new CustomerDto();
  14. customerDto.setCustomerName( customer.getName() );
  15. customerDto.setId( customer.getId() );
  16. return customerDto;
  17. }
  18. }

10、自定义类型转换

有时候,在对象转换的时候可能会出现这样一个问题,就是源对象中的类型是Boolean类型,而目标对象类型是String类型,这种情况可以通过@Mapper的uses属性来实现:

  1. @Data
  2. @NoArgsConstructor
  3. @AllArgsConstructor
  4. public class Customer {
  5. private Long id;
  6. private String name;
  7. private Boolean isDisable;
  8. }
  9. @Data
  10. public class CustomerDto {
  11. private Long id;
  12. private String customerName;
  13. private String disable;
  14. }

定义转换规则的类:

  1. public class BooleanStrFormat {
  2. public String toStr(Boolean isDisable) {
  3. if (isDisable) {
  4. return "Y";
  5. } else {
  6. return "N";
  7. }
  8. }
  9. public Boolean toBoolean(String str) {
  10. if (str.equals("Y")) {
  11. return true;
  12. } else {
  13. return false;
  14. }
  15. }
  16. }

定义Mapper,@Mapper( uses = { BooleanStrFormat.class}),注意,这里的users属性用于引用之前定义的转换规则的类:

  1. @Mapper( uses = { BooleanStrFormat.class})
  2. public interface CustomerMapper {
  3. CustomerMapper INSTANCES = Mappers.getMapper(CustomerMapper.class);
  4. @Mappings({
  5. @Mapping(source = "name", target = "customerName"),
  6. @Mapping(source = "isDisable", target = "disable")
  7. })
  8. CustomerDto toCustomerDto(Customer customer);
  9. }

这样子,Customer类中的isDisable属性的true就会转变成CustomerDto中的disable属性的yes。
MapStruct自动生成的类中的代码:

  1. @Generated(
  2. value = "org.mapstruct.ap.MappingProcessor",
  3. date = "2019-02-14T16:49:18+0800",
  4. comments = "version: 1.3.0.Final, compiler: javac, environment: Java 1.8.0_181 (Oracle Corporation)"
  5. )
  6. public class CustomerMapperImpl implements CustomerMapper {
  7. // 引用 uses 中指定的类
  8. private final BooleanStrFormat booleanStrFormat = new BooleanStrFormat();
  9. @Override
  10. public CustomerDto toCustomerDto(Customer customer) {
  11. if ( customer == null ) {
  12. return null;
  13. }
  14. CustomerDto customerDto = new CustomerDto();
  15. // 转换方式的使用
  16. customerDto.setDisable( booleanStrFormat.toStr( customer.getIsDisable() ) );
  17. customerDto.setCustomerName( customer.getName() );
  18. customerDto.setId( customer.getId() );
  19. return customerDto;
  20. }
  21. }

要注意的是,如果使用了例如像Spring这样的环境,Mapper引入uses类实例的方式将是自动注入,那么这个类也应该纳入Spring容器:
CustomerMapper.java指定使用Spring

  1. @Mapper(componentModel = "spring", uses = { BooleanStrFormat.class})
  2. public interface CustomerMapper {
  3. CustomerMapper INSTANCES = Mappers.getMapper(CustomerMapper.class);
  4. @Mappings({
  5. @Mapping(source = "name", target = "customerName"),
  6. @Mapping(source = "isDisable", target = "disable")
  7. })
  8. CustomerDto toCustomerDto(Customer customer);
  9. }

转换类要加入Spring容器:

  1. @Component
  2. public class BooleanStrFormat {
  3. public String toStr(Boolean isDisable) {
  4. if (isDisable) {
  5. return "Y";
  6. } else {
  7. return "N";
  8. }
  9. }
  10. public Boolean toBoolean(String str) {
  11. if (str.equals("Y")) {
  12. return true;
  13. } else {
  14. return false;
  15. }
  16. }
  17. }

MapStruct自动生成的类:

  1. @Generated(
  2. value = "org.mapstruct.ap.MappingProcessor",
  3. date = "2019-02-14T16:55:35+0800",
  4. comments = "version: 1.3.0.Final, compiler: javac, environment: Java 1.8.0_181 (Oracle Corporation)"
  5. )
  6. @Component
  7. public class CustomerMapperImpl implements CustomerMapper {
  8. // 使用自动注入的方式引入
  9. @Autowired
  10. private BooleanStrFormat booleanStrFormat;
  11. @Override
  12. public CustomerDto toCustomerDto(Customer customer) {
  13. if ( customer == null ) {
  14. return null;
  15. }
  16. CustomerDto customerDto = new CustomerDto();
  17. customerDto.setDisable( booleanStrFormat.toStr( customer.getIsDisable() ) );
  18. customerDto.setCustomerName( customer.getName() );
  19. customerDto.setId( customer.getId() );
  20. return customerDto;
  21. }
  22. }