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. java连接sql server 2008

    请先确保已经设置好了sa,如果不是,可以参照下面链接修改http://jingyan.baidu.com/article/8cdccae9452b3c315513cd52.html 然后重启数据库,重 ...

  2. 基于IntelliJ IDEA的代码评审插件 Code Review Plugin

    一.阿里规范公约 1.左上角 File -> Settings -> Plugins -> 搜索:Alibaba Java Coding Guidelines,安装插件并重启IDEA ...

  3. Day1学习总结

    # 1.print()# 2.input()# 3.if:# elif# else#4.while循环#5.for i in range()#6.break.continue#7.import ran ...

  4. 新闻网大数据实时分析可视化系统项目——12、Hive与HBase集成进行数据分析

    (一)Hive 概述 (二)Hive在Hadoop生态圈中的位置 (三)Hive 架构设计 (四)Hive 的优点及应用场景 (五)Hive 的下载和安装部署 1.Hive 下载 Apache版本的H ...

  5. axios设置请求头失效的问题

    前言:因为在使用vue-element-admin框架时遇到了设置请求头失效的问题,在后来发现是代理跨域问题,所以又简单理解了一下跨域. 出现的问题是我在axios拦截器上设置了请求头token,但是 ...

  6. JS 函数创建、封装、调用

    一.简单函数创建.封装 第三种就是构造函数 function fun(a,b){ this.firstName=a this.lastName=b } var x=new myFun(Jhon,Dav ...

  7. 【android官方文档】与其他App交互

    发送用户到另外一个App YOU SHOULD ALSO READ 内容分享 One of Android's most important features is an app's ability ...

  8. Core Data 基本数据操作 增删改查 排序

    所有操作都基于Core Data框架相关 API,工程需要添加CoreData.framework支持 1.增  NSEntityDescription insertNewObjectForEntit ...

  9. 循环调用spring的dao,数个过后无响应

    循环调用spring的dao,数个过后无响应 博客分类: spring daospringssh      最近遇到这么一个问题:前台按钮发送AJax请求到后台,后台是SSH框架.每点击一下按钮就发送 ...

  10. ADS 2015破解方法--笔记

    关键步骤是:设定系统环境变量HPEESOF_LIC_DIR,然后重启,再打开License Manager进行License的添加