遇到MapStruct后,再也不手写PO,DTO,VO对象之间的转换了

在这里插入图片描述

介绍

在工作中,我们经常要进行各种对象之间的转换。

PO:persistent object 持久对象,对应数据库中的一条记录
VO:view object 表现层对象,最终返回给前端的对象
DTO:data transfer object数据传输对象,如dubbo服务之间传输的对象

如果这些对象的属性名相同还好,可以用如下工具类赋值

Spring BeanUtils
Cglib BeanCopier
避免使用Apache BeanUtils,性能较差

如果属性名不同呢?如果是将多个PO对象合并成一个VO对象呢?好在有MapStruct神器,可以帮助我们快速转换

在pom文件中加入如下依赖即可

<dependency>
    <groupId>org.mapstruct</groupId>
    <artifactId>mapstruct-jdk8</artifactId>
    <version>1.2.0.CR1</version>
</dependency>

<dependency>
    <groupId>org.mapstruct</groupId>
    <artifactId>mapstruct-processor</artifactId>
    <version>1.2.0.CR1</version>
    <scope>provided</scope>
</dependency>

对象互转

@Data
@Builder
public class StudentPO {

    private Integer id;
    private String name;
    private Integer age;
    private String className;
}
@Data
public class StudentVO {

    private Integer id;
    private String studentName;
    private Integer studentAge;
    private String schoolName;
}
@Mapper
public interface StudentMapper {

    StudentMapper INSTANCE = Mappers.getMapper(StudentMapper.class);

    @Mappings({
            @Mapping(source = "name", target = "studentName"),
            @Mapping(source = "age", target = "studentAge")
    })
    StudentVO po2Vo(StudentPO studentPO);
}
  1. 新建一个Mapper接口,上面加上@Mapper注解
  2. 新建一个成员变量INSTANCE
  3. 用@Mapping注解指定映射关系,名字相同的就不用再指定了,会自动映射

测试效果如下,名字不同且没有指定映射关系的会被设置为null

@Test
public void studentPo2Vo() {
    StudentPO studentPO = StudentPO.builder().id(10).name("test")
            .age(24).className("教室名").build();
    StudentVO studentVO = StudentMapper.INSTANCE.po2Vo(studentPO);
    // StudentVO(id=10, studentName=test, studentAge=24, schoolName=null)
    System.out.println(studentVO);
}

List互转

@Mapper
public interface StudentMapper {

    StudentMapper INSTANCE = Mappers.getMapper(StudentMapper.class);

    @Mappings({
            @Mapping(source = "name", target = "studentName"),
            @Mapping(source = "age", target = "studentAge")
    })
    StudentVO po2Vo(StudentPO studentPO);

    List<StudentVO> poList2VoList(List<StudentPO> studentPO);
}

List类型互转的映射规则会用单个对象的映射规则,看测试效果

@Test
public void poList2VoList() {
    List<StudentPO> studentPOList = new ArrayList<>();
    for (int i = 1; i <= 2; i++) {
        StudentPO studentPO = StudentPO.builder().id(i).name(String.valueOf(i)).age(i).build();
        studentPOList.add(studentPO);
    }
    List<StudentVO> studentVOList = StudentMapper.INSTANCE.poList2VoList(studentPOList);
    // [StudentVO(id=1, studentName=1, studentAge=1, schoolName=null),
    // StudentVO(id=2, studentName=2, studentAge=2, schoolName=null)]
    System.out.println(studentVOList);
}

多个对象映射一个对象

我们用SchoolPO和StudentPO来映射SchoolStudentVO

@Data
@Builder
public class SchoolPO {

    private String name;
    private String location;
}
@Data
@Builder
public class StudentPO {

    private Integer id;
    private String name;
    private Integer age;
    private String className;
}
@Data
public class SchoolStudentVO {

    private String schoolName;
    private String studentName;
}
@Mapper
public interface StudentMapper {

    @Mappings({
            @Mapping(source = "schoolPO.name", target = "schoolName"),
            @Mapping(source = "studentPO.name", target = "studentName")
    })
    SchoolStudentVO mergeVo(SchoolPO schoolPO, StudentPO studentPO);
}

测试例子如下

@Test
public void mergeVo() {
    SchoolPO schoolPO = SchoolPO.builder().name("学校名字").build();
    StudentPO studentPO = StudentPO.builder().name("学生名字").build();
    SchoolStudentVO schoolStudentVO = StudentMapper.INSTANCE.mergeVo(schoolPO, studentPO);
    // SchoolStudentVO(schoolName=学校名字, studentName=学生名字)
    System.out.println(schoolStudentVO);
}

当然还有其他的骚操作,这里就简单介绍一个比较实用的技巧,有兴趣的可以看官方的example

https://github.com/mapstruct/mapstruct-examples

转换的时候指定方法

@Data
@Builder
public class PersonVO {

    private int age;
    private GenderEnum gender;
}
@Getter
@AllArgsConstructor
public enum GenderEnum {

    FEMALE(0, "女性"),
    MALE(1, "男性");

    private int value;
    private String name;

}
@Data
public class PersonPO {

    private int age;
    private int gender;
}

当我们想把PersonVO转换为PersonPO,PersonPO中的gender是一个枚举,PersonVO中的gender是一个int类型,如何把枚举转为枚举的value值呢?

在@Mapping注解中通过qualifiedByName属性指定转换的方法

@Mapper
public interface StudentMapper {

    StudentMapper INSTANCE = Mappers.getMapper(StudentMapper.class);

    @Mappings({
            @Mapping(source = "gender", target = "gender", qualifiedByName = "convertGender"),
    })
    PersonPO vo2Po(PersonVO personVO);

    @Named("convertGender")
    default Integer convertTargetType(GenderEnum genderEnum) {
        return genderEnum.getValue();
    }

}

测试类如下

@Test
public void specifyMethod() {
    PersonVO personVO = PersonVO.builder().age(10).gender(GenderEnum.MALE).build();
    PersonPO personPO = StudentMapper.INSTANCE.vo2Po(personVO);
    // PersonPO(age=10, gender=1)
    System.out.println(personPO);
}

实现原理

MapStruct帮你对接口生成了一个实现类,下面就是生成的实现类,从class文件夹中可以看到

public class StudentMapperImpl implements StudentMapper {

    @Override

    public StudentVO po2Vo(StudentPO studentPO) {

        if ( studentPO == null ) {

            return null;
        }

        StudentVO studentVO = new StudentVO();

        studentVO.setStudentAge( studentPO.getAge() );

        studentVO.setStudentName( studentPO.getName() );

        studentVO.setId( studentPO.getId() );

        return studentVO;
    }

    @Override

    public List<StudentVO> poList2VoList(List<StudentPO> studentPO) {

        if ( studentPO == null ) {

            return null;
        }

        List<StudentVO> list = new ArrayList<StudentVO>( studentPO.size() );

        for ( StudentPO studentPO1 : studentPO ) {

            list.add( po2Vo( studentPO1 ) );
        }

        return list;
    }

    @Override

    public SchoolStudentVO mergeVo(SchoolPO schoolPO, StudentPO studentPO) {

        if ( schoolPO == null && studentPO == null ) {

            return null;
        }

        SchoolStudentVO schoolStudentVO = new SchoolStudentVO();

        if ( schoolPO != null ) {

            schoolStudentVO.setSchoolName( schoolPO.getName() );
        }

        if ( studentPO != null ) {

            schoolStudentVO.setStudentName( studentPO.getName() );
        }

        return schoolStudentVO;
    }

    @Override

    public PersonPO vo2Po(PersonVO personVO) {

        if ( personVO == null ) {

            return null;
        }

        PersonPO personPO = new PersonPO();

        personPO.setGender( convertTargetType( personVO.getGender() ) );

        personPO.setAge( personVO.getAge() );

        return personPO;
    }
}

欢迎关注

在这里插入图片描述

参考博客

[0]http://www.tianshouzhi.com/api/tutorials/mapstruct
[1]https://www.jianshu.com/p/3f20ca1a93b0
[2]https://juejin.im/entry/5b228c2651882574b15882ba
各种O的区别
[3]https://blog.csdn.net/u011870547/article/details/81077153

  • 5
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Java识堂

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值