1. Retrieving a mapper(检索映射器)

4.1. The Mappers factory(映射工厂)

可以通过 org.mapstruct.factory.Mappers 类检索映射器实例。只需调用 getMapper () 方法, 传递映射器的接口类型即可返回:

Example 15. Using the Mappers factory

CarMapper mapper = Mappers.getMapper( CarMapper.class );

按照约定, 映射器接口应定义一个名为【INSTANCE】的成员, 【INSTANCE】包含映射器类型的单个实例:

Example 16. Declaring an instance of a mapper

@Mapper
public interface CarMapper { CarMapper INSTANCE = Mappers.getMapper( CarMapper.class ); CarDto carToCarDto(Car car);
}

这种模式使得客户端可以很容易地使用映射器对象, 而不必反复实例化新实例:

Example 17. Accessing a mapper

Car car = ...;
CarDto dto = CarMapper.INSTANCE.carToCarDto( car );

请注意, 由 MapStruct 生成的映射程序是线程安全的, 因此多个线程可以安全地同时访问。

4.2. Using dependency injection(使用依赖项注入)

如果您使用的是依赖注入框架 (如 CDI (JavaTM EE 的上下文和依赖项注入) 或 Spring 框架, 则建议通过依赖项注入获取映射器对象。为此, 您可以指定组件模型,那些生成的映射器类应基于通过@Mapper#componentModel或使用处理器选项中的配置选项。

目前有支持的 CDI 和Spring的依赖注入。

Example 18. A mapper using the CDI component model

@Mapper(componentModel = "cdi")
public interface CarMapper { CarDto carToCarDto(Car car);
}
  1. Data type conversions(数据类型转换)

在源和目标对象中, 映射属性不总是具有相同的类型。例如, 属性可以是源 bean 中的 int 类型, 而在目标 bean 中为 Long 类型。

另一个示例是对应映射到目标模型中相应类型的其他对象的引用。类Car可能有一个属性driver且类型是Person,它 需要转换成一个 PersonDto 对象时。

在本节中, 您将了解 MapStruct 如何处理此类数据类型转换。

5.1. Implicit type conversions(隐式类型转换)

MapStruct 在许多情况下自动处理类型转换。例如, 如果属性在源 bean 中是 int 类型, 而在目标 bean 中是类型String,那么代码将分别调用String#valueOf(int) 和 Integer#parseInt(String)来转换类型。以下几种情形,MapStruct将执行自动转换:

  • 在所有 Java 基本数据类型及其相应的包装类型之间。例如:int和Integer;long和Long等。当将包装类型转换为相应的基元类型时, 将执行 null 检查。
  • 在所有 Java 基本数字类型和包装类型之间。例如:int 和long ; byte 和 Integer.
  • 在所有 Java 基本类型 (包括它们的包装) 和String之间。例如:int和String;Integer和String;等。可以指定 转换成java.text.DecimalFormat 所理解的格式字符串。

Example 20. Conversion from int to String

@Mapper
public interface CarMapper { @Mapping(source = "price", numberFormat = "$#.00")
CarDto carToCarDto(Car car); @IterableMapping(numberFormat = "$#.00")
List<String> prices(List<Integer> prices);
}
  • 在枚举类型和字符串之间。
  • 大数字类型(java.math.BigInteger, java.math.BigDecimal)和基本数字类型(包括它们的包装类型)。可以指定 转换成java.text.DecimalFormat 所理解的格式字符串。
  • 大数字类型(java.math.BigInteger, java.math.BigDecimal)和String。可以指定 转换成java.text.DecimalFormat 所理解的格式字符串。

Example 21. Conversion from BigDecimal to String

@Mapper
public interface CarMapper { @Mapping(source = "power", numberFormat = "#.##E0")
CarDto carToCarDto(Car car); }

以下不再翻译

  • Between JAXBElement and T, List<JAXBElement> and List

  • Between java.util.Calendar/java.util.Date and JAXB’s XMLGregorianCalendar

  • Between java.util.Date/XMLGregorianCalendar and String. A format string as understood by java.text.SimpleDateFormat can be specified via the dateFormat option as this:Example 22. Conversion from Date to String

    @Mapper

    public interface CarMapper {

        @Mapping(source = "manufacturingDate", dateFormat = "dd.MM.yyyy")
    CarDto carToCarDto(Car car); @IterableMapping(dateFormat = "dd.MM.yyyy")
    List<String> stringListToDateList(List<Date> dates);
    }
    • Between Jodas org.joda.time.DateTime, org.joda.time.LocalDateTime, org.joda.time.LocalDate, org.joda.time.LocalTime and String. A format string as understood by java.text.SimpleDateFormat can be specified via the dateFormat option (see above).
    • Between Jodas org.joda.time.DateTime and javax.xml.datatype.XMLGregorianCalendar, java.util.Calendar.
    • Between Jodas org.joda.time.LocalDateTime, org.joda.time.LocalDate and javax.xml.datatype.XMLGregorianCalendar, java.util.Date.
    • Between java.time.ZonedDateTime, java.time.LocalDateTime, java.time.LocalDate, java.time.LocalTime from Java 8 Date-Time package and String. A format string as understood by java.text.SimpleDateFormat can be specified via the dateFormat option (see above).
    • Between java.time.ZonedDateTime from Java 8 Date-Time package and java.util.Date where, when mapping a ZonedDateTime from a given Date, the system default timezone is used.
    • Between java.time.LocalDateTime from Java 8 Date-Time package and java.util.Date where timezone UTC is used as the timezone.
    • Between java.time.LocalDate from Java 8 Date-Time package and java.util.Date where timezone UTC is used as the timezone.
    • Between java.time.ZonedDateTime from Java 8 Date-Time package and java.util.Calendar.
    • Between java.sql.Date and java.util.Date
    • Between java.sql.Time and java.util.Date
    • Between java.sql.Timestamp and java.util.Date
    • When converting from a String, omitting Mapping#dateFormat, it leads to usage of the default pattern and date format symbols for the default locale. An exception to this rule is XmlGregorianCalendar which results in parsing the String according to XML Schema 1.0 Part 2, Section 3.2.7-14.1, Lexical Representation.

5.2. Mapping object references(映射对象引用)

通常,一个对象的属性不仅有基本类型还有引用类型。例如:Car类中有个类型为Person的driver属性,当Car映射为CarDto时,driver映射到目标对象上后,应该是PersonDto。在这种情况下, 只需定义引用对象类型的映射方法:

Example 23. Mapper with one mapping method using another

@Mapper
public interface CarMapper { CarDto carToCarDto(Car car); PersonDto personToPersonDto(Person person);
}

这样,在生成的代码中,在driver需要转换时,会利用personToPersonDto()方法,把Car中的Person转换成CarDto中的PersonDto。

这样就可以映射任意深度的对象。当从实体映射到数据传输对象时, 在某个点上的引用对象将被继续细分映射。

在生成映射方法的实现时, MapStruct 将对源对象和目标对象中的每个属性对应用以下规则:

  • 如果目标属性与源属性是同一个类型,则进行简单复制即可。
  • 如果目标属性与源属性不是同一个类型,则检测是否有声明的子映射方法,如果存在,则使用子映射方法。
  • 如果不存在此类方法, MapStruct 将查看该属性的源和目标类型的内置转换是否存在。如果存在, 生成的映射代码将应用此转换。
  • 如果找不到此类方法, MapStruct 将尝试生成一个自动子映射方法, 将在源和目标属性之间进行映射。
  • 如果 MapStruct 无法创建基于名称的映射方法, 则在生成时将引发错误, 指示非可映射属性及其路径。

为了阻止 MapStruct 生成自动的子映射方法,可以使用@Mapper( disableSubMappingMethodsGeneration = true ).

5.3. Controlling nested bean mappings(控制嵌套 bean 映射)

如上文所述, MapStruct 将根据源和目标属性的名称生成方法。不幸的是, 在许多场合, 这些名字并不匹配。@Mapping 源或目标类型中的‘.’ 表示法可用于控制当名称不匹配时如何映射属性。在我们的示例存储库中有一个很详细的示例, 可以解释如何克服这个问题。

Example 24. Mapper controlling nested beans mappings I

@Mapper
public interface FishTankMapper { @Mappings({
@Mapping(target = "fish.kind", source = "fish.type"),
@Mapping(target = "fish.name", ignore = true),
@Mapping(target = "ornament", source = "interior.ornament"),
@Mapping(target = "material.materialType", source = "material"),
@Mapping(target = "quality.report.organisation.name", source = "quality.report.organisationName")
})
FishTankDto map( FishTank source );
}

4. Retrieving a mapper(检索映射器)的更多相关文章

  1. kali之Nmap (Network Mapper(网络映射器)

    Nmap是主机扫描工具,他的图形化界面是Zenmap,分布式框架为Dnamp. Nmap可以完成以下任务: 主机探测 端口扫描 版本检测 系统检测 支持探测脚本的编写 Nmap在实际中应用场合如下: ...

  2. Mybatis映射器接口代理对象的方式 运行过程

    查询一张表的所有数据. 环境: 使用工具IntelliJ IDEA 2018.2版本. 创建Maven工程不用骨架 1.pom.xml <?xml version="1.0" ...

  3. Mybatis 映射器接口实现类的方式 运行过程debug分析

    查询一张表的所有数据. 环境: 使用工具IntelliJ IDEA 2018.2版本. 创建Maven工程不用骨架 <?xml version="1.0" encoding= ...

  4. 带码农《手写Mybatis》进度3:实现映射器的注册和使用

    作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获!

  5. Mapper映射器

    在两个独立的对象之间建立通信的对象 需要在两个必须相互隔离的子系统间建立通信. 可能是因为无法修改已有的子系统,或者不愿意在两者之间建立依赖关系.甚至不愿意这两个子系统与另一个部件间建立依赖关系. 运 ...

  6. MyBatis中映射器Mapper概述

    MyBatis真正强大之处在于它的映射器.因为它异常强大并且编写相对简单,不仅比传统编写SQL语句做的更好并且能节省将近95%的代码量 XML中顶级元素汇总 cache: 给定命名空间的缓存配置 ca ...

  7. Spring集成MyBatis的使用-使用Mapper映射器

    Spring集成MyBatis使用 前面复习MyBatis时,发现在测试时,需要手动创建sqlSessionFactory,Spring将帮忙自动创建sqlSessionFactory,并且将自动扫描 ...

  8. MyBatis数据库连接的基本使用-补充Mapper映射器

    补充 Mapper映射器的使用: Mapper映射器,google添加.Mapper映射器是将mapper.xml中配置的sql id,parameterType和resultMap按照规则一一映射到 ...

  9. DART: a fast and accurate RNA-seq mapper with a partitioning strategy DART:使用分区策略的快速准确的RNA-seq映射器

    DART: a fast and accurate RNA-seq mapper with a partitioning strategyDART:使用分区策略的快速准确的RNA-seq映射器 Abs ...

随机推荐

  1. 本周总结(19年暑假)—— Part3

    日期:2019.7.28 博客期:109 星期日 这几天要练车,嗯呢,但是对于分布式数据库的研究并没有停止!

  2. 编写安全 PHP 应用程序的七个习惯

    编写安全 PHP 应用程序的七个习惯   在提及安全性问题时,需要注意,除了实际的平台和操作系统安全性问题之外,您还需要确保编写安全的应用程序.在编写 PHP 应用程序时,请应用下面的七个习惯以确保应 ...

  3. JAVA 内部类 总结

    内部类是指在一个外部类的内部再定义一个类.内部类作为外部类的一个成员,并且依附于外部类而存在的.内部类可为静态,可用protected和private修饰(而外部类只能使用public和缺省的包访问权 ...

  4. UIWindow的获取

    注意:还是直接用下面这个比较靠谱.尤其是iOS11之后. [UIApplication sharedApplication].keyWindow;   1.下面这种是比较严谨的方式 - (UIWind ...

  5. 图片IO域 旋转画面的组态 图片是4个静止的风扇 PLC的MW6为风扇指针..

    图片IO域 旋转画面的组态 图片是4个静止的风扇 PLC的MW6为风扇指针.. Plc在循环中断组织块 OB35 中 将MW6 每100ms 加1 加到4 清0 图片[MW6] MW6 是图片指针 对 ...

  6. JSON对象和字符串的收发(JS客户端用typeof()进行判断非常重要)

    Ajax前台向后台传递对象: 数据准备 将js对象或者json对象转换为json字符串在Ajax传递,在后台中再将json字符串转换为json对象,再转换为java对象或在前端和后端构造一样的数据结构 ...

  7. mabatis--使用mapper代理开发dao

    1.编写mapper.xml映射文件: <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE m ...

  8. 「CF521D」Shop

    传送门 Luogu 解题思路 当只有第三类操作时,我们显然先进行val较大的操作,这是显然的. 那么就考虑把所有的操作都转变为第三类操作. 第一类操作,显然很容易变为第二类操作:单点维护最大的最终结果 ...

  9. 068、Java面向对象之声明两个对象

    01.代码如下: package TIANPAN; class Book { // 定义一个新的类 String title; // 书的名字 double price; // 书的价格 public ...

  10. 防火墙、WAF、IPS、IDS都是什么

    防火墙 (Firewall) 别名防护墙,于1993发明并引入国际互联网. 他是一项信息安全的防护系统,依照特定的规则,允许或是限制传输的数据通过.在网络中,所谓的防火墙是指一种将内网和外网分开的方法 ...