MapStruct

MapStruct

一.MapStruct是什么

MapStruct 是一个 Java注释处理器,用于生成类型安全的 bean 映射类。

MapStruct 的一般理念是生成的代码看起来尽可能像您自己亲手编写的代码。特别是这意味着通过简单的 getter/setter 调用而不是反射或类似方法将值从源复制到目标。

二.为什么要用MapStruct

类型安全、高性能和无依赖的 bean 映射代码

与动态映射框架相比,MapStruct 具有以下优势:

  • 通过使用普通方法调用而不是反射来快速执行
  • 编译时类型安全:只能映射相互映射的对象和属性,不能将订单实体意外映射到客户 DTO 等。
  • 在构建时清除错误报告,如果
    • 映射不完整(并非所有目标属性都被映射)
    • 映射不正确(找不到合适的映射方法或类型转换)

三.MapStruct怎么用

所以要做的就是定义一个映射器接口,该接口声明任何所需的映射方法。在编译期间,MapStruct 将生成此接口的实现。此实现使用纯 Java 方法调用来映射源对象和目标对象,即没有反射或类似的。

它包括以下工件:

  • org.mapstruct:mapstruct:包含所需的注释,例如@Mapping
  • org.mapstruct:mapstruct-processor:包含生成映射器实现的注释处理器

3.1 maven配置

对于基于 Maven 的项目,将以下内容添加到您的 POM 文件中以使用 MapStruct:(特别注明:lombok请使用1.16.16及以上版本)

<properties>
    <org.mapstruct.version>1.4.2.Final</org.mapstruct.version>
</properties>
...
<dependencies>
    <dependency>
            <groupId>org.mapstruct</groupId>
            <artifactId>mapstruct</artifactId>
            <version>${org.mapstruct.version}</version>
        </dependency>

        <dependency>
            <groupId>org.mapstruct</groupId>
            <artifactId>mapstruct-processor</artifactId>
            <version>${org.mapstruct.version}</version>
        </dependency>
</dependencies>
...

3.2 关键注解

@Mapper
@Mappings
@Mapping
@MappingTarget
@InheritInverseConfiguration
@IterableMapping

Mapping注解解析

public @interface Mapping {
    java.lang.String target();       // 目标

    java.lang.String source() default "";  //源

    java.lang.String dateFormat() default ""; //时间类型的日期格式

    java.lang.String numberFormat() default ""; //数字格式

    java.lang.String constant() default "";  //常量 源目标也必须要有对应字段,与defaultValue只能存一

    java.lang.String expression() default ""; //表达式

    java.lang.String defaultExpression() default "";

    boolean ignore() default false; //忽略 

    java.lang.Class<? extends java.lang.annotation.Annotation>[] qualifiedBy() default {};

    java.lang.String[] qualifiedByName() default {};

    java.lang.Class<?> resultType() default void.class;

    java.lang.String[] dependsOn() default {};

    java.lang.String defaultValue() default ""; // 来源的默认值,与constant只能存一,

    org.mapstruct.NullValueCheckStrategy nullValueCheckStrategy() default org.mapstruct.NullValueCheckStrategy.ON_IMPLICIT_CONVERSION;

    org.mapstruct.NullValuePropertyMappingStrategy nullValuePropertyMappingStrategy() default org.mapstruct.NullValuePropertyMappingStrategy.SET_TO_NULL;

    java.lang.Class<? extends java.lang.annotation.Annotation> mappingControl() default org.mapstruct.control.MappingControl.class;
}

3.3配置映射器

@Mapper
public interface WmsMapper {

    /**
     * 获取该类自动生成的实现类的实例
     * 接口中的属性都是 public static final 的 方法都是public abstract的
     */
    WmsMapper INSTANCES = Mappers.getMapper(WmsMapper.class);

}

3.4开始使用MapStruct

1.定义相互转化的实体

wms

@Data
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class Wms {

    private int id;

    private String name;
}
@Data
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class WmsDto {

    private int id;

    private String name;
}

2.简单映射方法

第一种:

在WmsMapper映射器里新增方法:

    /**
     * 对象映射
     * @param wms
     * @return
     */
    WmsDto wmsToWmsDto(Wms wms);

    /**
     * 批量映射
     * @param list
     * @return
     */
    List<WmsDto> wmsToWmsDto(List<Wms> list);

    /**
     * 对象映射反转 (当对象存在多个映射方法时,逆转方法需在注解中声明逆转的哪个方法,所以为了防止编译不通过,最好写的时候就在注解的name属性里声明好)
     * @param wmsDto
     * @return
     */
    @InheritInverseConfiguration(name = "wmsToWmsDto")
    Wms wmsDtoToWms(WmsDto wmsDto);

    /**
     * 批量映射 (当对象存在多个映射方法时,逆转方法需在注解中声明逆转的哪个方法,所以为了防止编译不通过,最好写的时候就在注解的name属性里声明好)
     * @param list
     * @return
     */
    @InheritInverseConfiguration(name = "wmsToWmsDto")
    List<Wms> wmsDtoToWms(List<WmsDto> list);

测试代码:

	public static void main(String[] args) {

        System.out.println("MapStruct简单映射方法1------------start");

        Wms wms = Wms.builder().id(1).name("wms").build();
        WmsDto wmsToWmsDto = WmsMapper.INSTANCES.wmsToWmsDto(wms);
        System.out.println("对象映射结果"+wmsToWmsDto);

        System.out.println("批量对象映射结果---");
        WmsMapper.INSTANCES.wmsToWmsDto(Arrays.asList(wms,wms,wms)).forEach(System.out::println);

        System.out.println("映射反转---");
        Wms wmsDtoToWms = WmsMapper.INSTANCES.wmsDtoToWms(wmsToWmsDto);
        System.out.println("映射反转结果"+wmsDtoToWms);

        System.out.println("批量映射反转结果---");
        WmsMapper.INSTANCES.wmsDtoToWms(Arrays.asList(wmsToWmsDto,wmsToWmsDto,wmsToWmsDto)).forEach(System.out::println);

        System.out.println("MapStruct简单映射方法1------------end");
    }

运行结果:

MapStruct简单映射方法1------------start
对象映射结果WmsDto(id=1, name=wms)
批量对象映射结果---
WmsDto(id=1, name=wms)
WmsDto(id=1, name=wms)
WmsDto(id=1, name=wms)
映射反转---
映射反转结果Wms(id=1, name=wms)
批量映射反转结果---
Wms(id=1, name=wms)
Wms(id=1, name=wms)
Wms(id=1, name=wms)
MapStruct简单映射方法1------------end
第二种:

在WmsMapper映射器里新增方法:

	/**
     *  对象映射
     * @param wms
     * @param wmsDto
     */
    void wmsToWmsDtoMothod(Wms wms, @MappingTarget WmsDto wmsDto);
    
    /**
     *  批量对象映射
     * @param wms
     * @param wmsDto
     */
    void wmsToWmsDtoMothod(List<Wms> wms, @MappingTarget List<WmsDto> wmsDto);

测试代码:

	public static void main(String[] args) {

        System.out.println("MapStruct简单映射方法2***************start");

        Wms wms2 = Wms.builder().id(2).name("wms2").build();
        WmsDto wmsDto2 = WmsDto.builder().build();
        System.out.println("第二种映射前"+wmsDto2);
        WmsMapper.INSTANCES.wmsToWmsDtoMothod(wms2,wmsDto2);
        System.out.println("第二种映射后"+wmsDto2);
        
        List<Wms> WmsList = Arrays.asList(Wms.builder().id(1).name("wms1").build()
                ,Wms.builder().id(2).name("wms2").build()
                ,Wms.builder().id(3).name("wms3").build()
        );
        List<WmsDto> wmsDtoList = new ArrayList<>();
        WmsMapper.INSTANCES.wmsToWmsDtoMothod(WmsList,wmsDtoList);
        System.out.println("批量映射结果***************end");
        wmsDtoList.forEach(System.out::println);

        System.out.println("MapStruct简单映射方法2***************end");
    }

运行结果:

MapStruct简单映射方法2***************start
第二种映射前WmsDto(id=0, name=null)
第二种映射后WmsDto(id=2, name=wms2)
批量映射结果***************end
WmsDto(id=1, name=wms1)
WmsDto(id=2, name=wms2)
WmsDto(id=3, name=wms3)
MapStruct简单映射方法2***************end

3.不规则命名映射

在Wms实体新增age属性

private int age;

在WmsDto新增ageNumber属性

private int ageNumber;

在WmsMapper映射器里方法上加上映射规则:

	/**
     * 对象映射
     * @param wms
     * @return
     */
    @Mapping(source = "age" , target = "ageNumber")
    WmsDto wmsToWmsDto(Wms wms);
    
    /**
     *  对象映射
     * @param wms
     * @param wmsDto
     */
    @Mapping(source = "age" , target = "ageNumber")
    void wmsToWmsDtoMothod(Wms wms, @MappingTarget WmsDto wmsDto);

测试代码:

    public static void main(String[] args) {

        System.out.println("MapStruct简单映射方法1------------start");

        Wms wms = Wms.builder().id(1).name("wms").age(24).build();
        WmsDto wmsToWmsDto = WmsMapper.INSTANCES.wmsToWmsDto(wms);
        System.out.println("对象映射结果"+wmsToWmsDto);

        System.out.println("批量对象映射结果---");
        WmsMapper.INSTANCES.wmsToWmsDto(Arrays.asList(wms,wms,wms)).forEach(System.out::println);

        System.out.println("映射反转---");
        Wms wmsDtoToWms = WmsMapper.INSTANCES.wmsDtoToWms(wmsToWmsDto);
        System.out.println("映射反转结果"+wmsDtoToWms);

        System.out.println("批量映射反转结果---");
        WmsMapper.INSTANCES.wmsDtoToWms(Arrays.asList(wmsToWmsDto,wmsToWmsDto,wmsToWmsDto)).forEach(System.out::println);

        System.out.println("MapStruct简单映射方法1------------end");


        System.out.println("MapStruct简单映射方法2***************start");

        Wms wms2 = Wms.builder().id(2).name("wms2").age(10).build();
        WmsDto wmsDto2 = WmsDto.builder().build();
        System.out.println("第二种映射前"+wmsDto2);
        WmsMapper.INSTANCES.wmsToWmsDtoMothod(wms2,wmsDto2);
        System.out.println("第二种映射后"+wmsDto2);


        List<Wms> WmsList = Arrays.asList(Wms.builder().id(1).age(12).name("wms1").build()
                ,Wms.builder().id(2).name("wms2").age(12).build()
                ,Wms.builder().id(3).name("wms3").age(12).build()
        );
        List<WmsDto> wmsDtoList = new ArrayList<>();
        WmsMapper.INSTANCES.wmsToWmsDtoMothod(WmsList,wmsDtoList);
        System.out.println("批量映射结果***************");
        wmsDtoList.forEach(System.out::println);



        System.out.println("MapStruct简单映射方法2***************end");
    }

运行结果:

MapStruct简单映射方法1------------start
对象映射结果WmsDto(id=1, name=wms, ageNumber=24)
批量对象映射结果---
WmsDto(id=1, name=wms, ageNumber=24)
WmsDto(id=1, name=wms, ageNumber=24)
WmsDto(id=1, name=wms, ageNumber=24)
映射反转---
映射反转结果Wms(id=1, name=wms, age=24)
批量映射反转结果---
Wms(id=1, name=wms, age=24)
Wms(id=1, name=wms, age=24)
Wms(id=1, name=wms, age=24)
MapStruct简单映射方法1------------end
MapStruct简单映射方法2***************start
第二种映射前WmsDto(id=0, name=null, ageNumber=0)
第二种映射后WmsDto(id=2, name=wms2, ageNumber=10)
批量映射结果***************
WmsDto(id=1, name=wms1, ageNumber=12)
WmsDto(id=2, name=wms2, ageNumber=12)
WmsDto(id=3, name=wms3, ageNumber=12)
MapStruct简单映射方法2***************end

Process finished with exit code 0

注:只用在上述映射方法上声明即可,批量映射及映射反转不用声明

4.多对一映射

第一种多对象映射:

WmsDto对象新增属性

private String memberName;

private int myAge;

新增对象

@Data
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class Member {
    private String myName;
    private int myAge;
}

新增映射方法

@Mapping(source = "member.myAge" ,target = "myAge")
@Mapping(source = "member.myName" ,target = "memberName")
@Mapping(source = "wms.age" , target = "ageNumber")
WmsDto wmsAndMemberToWmsDto(Wms wms,Member member);

注:多对一映射时,与映射对象名一样的不用再次声明,源对象唯一命名不用加声明前缀,但是为了可读性,最好是加上声明前缀

测试代码

public static void main(String[] args) {
        Wms wms = Wms.builder().id(1).name("wms").age(2).build();
        Member member = Member.builder().myName("ssk").myAge(24).build();

        WmsDto wmsDto = WmsMapper.INSTANCES.wmsAndMemberToWmsDto(wms, member);
        System.out.println(wmsDto);
    }

运行结果

WmsDto(id=1, name=wms, ageNumber=2, memberName=ssk, myAge=24)

特殊:在多对一映射时,可以选择性忽略某些属性的映射

映射方法修改:

@Mapping(source = "member.myAge" ,target = "myAge")
    @Mapping(source = "member.myName" ,target = "memberName")
    @Mapping(source = "wms.age" , target = "ageNumber")
    @Mapping(target = "id" ,ignore = true)
    WmsDto wmsAndMemberToWmsDto(Wms wms,Member member);

运行结果

WmsDto(id=0, name=wms, ageNumber=2, memberName=师宋奎, myAge=24)
第二种常量映射:

新增映射方法

	@Mapping(source = "id",target = "id" )
    @Mapping(source = "name",target = "name" )
    WmsDto wmsAndMemberToWmsDto(Wms wms,String name,int id);

测试代码:

public static void main(String[] args) {
    Wms wms = Wms.builder().id(1).build();
    WmsDto wmsDto = WmsMapper.INSTANCES.wmsAndMemberToWmsDto(wms, "wms", 11);
    System.out.println(wmsDto);

}

运行结果

WmsDto(id=11, name=wms, ageNumber=0, memberName=null, myAge=0)

5.@Mapping(defaultValue = “wd” )

注:

声明 ignore = true 不生效

源属性有值 不生效

映射方法修改:

	@Mapping(source = "member.myAge" ,target = "myAge")
    @Mapping(source = "member.myName" ,target = "memberName")
    @Mapping(source = "wms.age" , target = "ageNumber")
    @Mapping(target = "id" ,ignore = true)
    @Mapping(target = "name" ,defaultValue = "wms" )
    WmsDto wmsAndMemberToWmsDto(Wms wms,Member member);

测试代码

public static void main(String[] args) {
        Wms wms = Wms.builder().id(1).build();
        WmsDto wmsDto = WmsMapper.INSTANCES.wmsAndMemberToWmsDto(wms, null);
        System.out.println(wmsDto);
    }

运行结果

WmsDto(id=0, name=wms, ageNumber=0, memberName=null, myAge=0)

6.@Mapping(constant= “11” )

注:

声明 ignore = true 不生效

源属性有值 替换

修改映射方法

@Mapping(target = “id” ,constant = “11”)

	@Mapping(source = "member.myAge" ,target = "myAge")
    @Mapping(source = "member.myName" ,target = "memberName")
    @Mapping(source = "wms.age" , target = "ageNumber")
    @Mapping(target = "id" ,constant = "11")
    @Mapping(target = "name" ,defaultValue = "wms" )
    WmsDto wmsAndMemberToWmsDto(Wms wms,Member member);

7.@Mapping(expression = “java(new Date())” )

WmsDto新增属性

private Date time;

修改映射方法

    @Mapping(source = "id",target = "id" )
    @Mapping(source = "name",target = "name" )
    @Mapping(target = "time" ,expression = "java(new java.util.Date())")
    WmsDto wmsAndMemberToWmsDto(Wms wms,String name,int id);

测试代码

public static void main(String[] args) {
		Wms wms = Wms.builder().id(1).build();
        WmsDto wmsDto1 = WmsMapper.INSTANCES.wmsAndMemberToWmsDto(wms, "wms", 11);
        System.out.println(wmsDto1);

    }

运行结果:

WmsDto(id=11, name=wms, ageNumber=0, memberName=null, myAge=0, time=Wed Sep 15 15:13:20 CST 2021)

其他

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-WkOUDBXJ-1631951526475)(C:\Users\edy\AppData\Roaming\Typora\typora-user-images\image-20210915172448682.png)]

修改方式

在映射器@Mapper注解里新增属性

@Mapper(imports = {Date.class})

方法上声明变为

	@Mapping(source = "id",target = "id" )
    @Mapping(source = "name",target = "name" )
    @Mapping(target = "time" ,expression = "java(new Date())")
    WmsDto wmsAndMemberToWmsDto(Wms wms,String name,int id);

8.从int到String的转换 numberFormat

新增对象

@Data
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class Base {
    private int price;
    private BigDecimal power;
    private Date time;
}
@Data
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class BaseDto {
    private String price;
    private String power;
    private String time;
}

在映射器里新增方法

	/**
     *从int到String的转换
     * @param
     * @return
     */
    @Mapping(target = "price", source = "price", numberFormat = "$#.00")
    BaseDto baseToBaseDTOShowInt(Base base);


    @IterableMapping(numberFormat = "$#.00")
    List<String> intToString(List<Integer> list);

    @InheritInverseConfiguration(name = "intToString")
    List<Integer> StringToInt(List<String> list);

测试代码

public static void main(String[] args) {
        Base base = Base.builder().price(100).build();
        BaseDto baseDto = WmsMapper.INSTANCES.baseToBaseDTOShowInt(base);
        System.out.println(baseDto);
        System.out.println("-------------分割线------------");
        List<String> strings = WmsMapper.INSTANCES.intToString(Arrays.asList(1, 2, 30, 40));
        strings.forEach(System.out::println);
        System.out.println("-------------分割线------------");
        WmsMapper.INSTANCES.StringToInt(strings).forEach(System.out::println);
    }

运行结果

BaseDto(price=$100.00, power=null, time=null)
-------------分割线------------
$1.00
$2.00
$30.00
$40.00
-------------分割线------------
1
2
30
40

9.从BigDecimal到String的转换 numberFormat

在映射器里新增方法

	/**
     *从BigDecimal到String的转换
     * @param
     * @return
     */
    @Mapping(target = "power", source = "power", numberFormat = "#.##E0")
    BaseDto baseToBaseDTOShowBigDecimal(Base base);

测试代码

public static void main(String[] args) {
        Base base = Base.builder().power(new BigDecimal(10.112)).build();
        BaseDto baseDto = WmsMapper.INSTANCES.baseToBaseDTOShowBigDecimal(base);
        System.out.println(baseDto);
    }

运行结果

BaseDto(price=0, power=1.01E1, time=null)

10.从日期到字符串的转换 dateFormat

在映射器里新增方法

/**
     *从日期到字符串的转换
     * @param
     * @return
     */
    @Mapping(target = "time", source = "time", dateFormat = "yyyy.MM.dd HH:mm:ss")
    BaseDto baseToBaseDTOShowDate(Base base);

    @IterableMapping(dateFormat = "dd.MM.yyyy")
    List<String> DateToString(List<Date> list);

    /**
     *   从字符串到日期的转换 逆转
     * @param list
     * @return
     */
    @InheritInverseConfiguration(name = "DateToString")
    List<Date> StringToDate(List<String> list);

测试代码

public static void main(String[] args) {
        System.out.println("------------baseToBaseDTOShowDate");
        Base base = Base.builder().time(new Date()).build();
        BaseDto baseDto = WmsMapper.INSTANCES.baseToBaseDTOShowDate(base);
        System.out.println(baseDto);

        System.out.println("------------DateToString");
        List<String> strings = WmsMapper.INSTANCES.DateToString(Arrays.asList(new Date(), new Date(), new Date(), new Date()));

        strings.forEach(System.out::println);

        System.out.println("------------StringToDate");
        WmsMapper.INSTANCES.StringToDate(strings).forEach(System.out::println);
    }

运行结果

------------baseToBaseDTOShowDate
BaseDto(price=0, power=null, time=2021.09.15 18:05:43)
------------DateToString
15.09.2021
15.09.2021
15.09.2021
15.09.2021
------------StringToDate
Wed Sep 15 00:00:00 CST 2021
Wed Sep 15 00:00:00 CST 2021
Wed Sep 15 00:00:00 CST 2021
Wed Sep 15 00:00:00 CST 2021

11.Map转换成对象

在映射器里新增方法

	@Mappings({
            @Mapping(expression = "java(Integer.valueOf(map.get(\"id\")))" ,target = "id",resultType = Integer.class),
            @Mapping(expression = "java(map.get(\"name\"))" ,target = "name",resultType = String.class),
            @Mapping(expression = "java(WmsMapMapper.stringToInteger((map.get(\"age\"))))" ,target = "age",resultType = 			Integer.class)
    })
    Wms mapToWms(Map<String,String> map);

    static Integer stringToInteger(String s){
        return Integer.valueOf(s);
    }

测试代码

public static void main(String[] args) {

        Map<String,String> map = new HashMap<>();
        map.put("id","1");
        map.put("name","ZSHMWMS组");
        map.put("age","2");

        Wms wms = WmsMapMapper.INSTANCES.mapToWms(map);
        System.out.println(wms);
    }

运行结果

Wms(id=1, name=ZSHMWMS组, age=2)

r.valueOf(map.get(“id”)))" ,target = “id”,resultType = Integer.class),
@Mapping(expression = “java(map.get(“name”))” ,target = “name”,resultType = String.class),
@Mapping(expression = “java(WmsMapMapper.stringToInteger((map.get(“age”))))” ,target = “age”,resultType = Integer.class)
})
Wms mapToWms(Map<String,String> map);

static Integer stringToInteger(String s){
    return Integer.valueOf(s);
}

测试代码

public static void main(String[] args) {

    Map<String,String> map = new HashMap<>();
    map.put("id","1");
    map.put("name","ZSHMWMS组");
    map.put("age","2");

    Wms wms = WmsMapMapper.INSTANCES.mapToWms(map);
    System.out.println(wms);
}

运行结果

Wms(id=1, name=ZSHMWMS组, age=2)

@MapperConfig
public interface BaseMapping<S, T> {

    /**
     * 映射同名属性
     *
     * @param source
     * @return
     */
    T sourceToTarget(S source);

    /**
     * 批量映射同名属性
     *
     * @param source
     * @return
     */
    @InheritConfiguration(name = "sourceToTarget")
    List<T> sourceToTarget(List<S> source);

    /**
     * 反向映射同名属性
     *
     * @param target
     * @return
     */
    @InheritInverseConfiguration(name = "sourceToTarget")
    S targetToSource(T target);

    /**
     * 批量反向映射同名属性
     *
     * @param target
     * @return
     */
    @InheritConfiguration(name = "targetToSource")
    List<S> targetToSource(List<T> target);

    /**
     * 流式批量映射同名属性
     *
     * @param source
     * @return
     */
    List<T> sourceToTarget(Stream<S> source);

    /**
     * 流式批量反向映射同名属性
     *
     * @param target
     * @return
     */
    @InheritConfiguration(name = "targetToSource")
    List<S> targetToSource(Stream<T> target);
}
上一篇:阿里云RDS如何进行账号及数据库管理?


下一篇:Mapstruct使用---一种JAVABEAN 转化工具