背景

mapStruct 是一个方便对象转换的工具,类似的工具还有 Dozer, BeanUtils。

实现

mapStruct的核心是在编译期生成基于转换规则的 Impl 文件,运行时直接调用 Impl 文件中的函数。整个 mapStruct 分成三个部分:

  1. 自定义注解,指定转换的规则。例如 source, target 等。

  2. freemarker 模板,用来生成 impl 文件。

  3. 基于 javax.annotation.processing 的处理模块。

基本流程是

#mermaid-1558415763951 .label{font-family:trebuchet ms,verdana,arial;color:#333}#mermaid-1558415763951 .node circle,#mermaid-1558415763951 .node ellipse,#mermaid-1558415763951 .node polygon,#mermaid-1558415763951 .node rect{fill:#ececff;stroke:#9370db;stroke-width:1px}#mermaid-1558415763951 .node.clickable{cursor:pointer}#mermaid-1558415763951 .arrowheadPath{fill:#333}#mermaid-1558415763951 .edgePath .path{stroke:#333;stroke-width:1.5px}#mermaid-1558415763951 .edgeLabel{background-color:#e8e8e8}#mermaid-1558415763951 .cluster rect{fill:#ffffde!important;stroke:#aa3!important;stroke-width:1px!important}#mermaid-1558415763951 .cluster text{fill:#333}#mermaid-1558415763951 div.mermaidTooltip{position:absolute;text-align:center;max-width:200px;padding:2px;font-family:trebuchet ms,verdana,arial;font-size:12px;background:#ffffde;border:1px solid #aa3;border-radius:2px;pointer-events:none;z-index:100}#mermaid-1558415763951 .actor{stroke:#ccf;fill:#ececff}#mermaid-1558415763951 text.actor{fill:#000;stroke:none}#mermaid-1558415763951 .actor-line{stroke:grey}#mermaid-1558415763951 .messageLine0{marker-end:"url(#arrowhead)"}#mermaid-1558415763951 .messageLine0,#mermaid-1558415763951 .messageLine1{stroke-width:1.5;stroke-dasharray:"2 2";stroke:#333}#mermaid-1558415763951 #arrowhead{fill:#333}#mermaid-1558415763951 #crosshead path{fill:#333!important;stroke:#333!important}#mermaid-1558415763951 .messageText{fill:#333;stroke:none}#mermaid-1558415763951 .labelBox{stroke:#ccf;fill:#ececff}#mermaid-1558415763951 .labelText,#mermaid-1558415763951 .loopText{fill:#000;stroke:none}#mermaid-1558415763951 .loopLine{stroke-width:2;stroke-dasharray:"2 2";marker-end:"url(#arrowhead)";stroke:#ccf}#mermaid-1558415763951 .note{stroke:#aa3;fill:#fff5ad}#mermaid-1558415763951 .noteText{fill:#000;stroke:none;font-family:trebuchet ms,verdana,arial;font-size:14px}#mermaid-1558415763951 .section{stroke:none;opacity:.2}#mermaid-1558415763951 .section0{fill:rgba(102,102,255,.49)}#mermaid-1558415763951 .section2{fill:#fff400}#mermaid-1558415763951 .section1,#mermaid-1558415763951 .section3{fill:#fff;opacity:.2}#mermaid-1558415763951 .sectionTitle0,#mermaid-1558415763951 .sectionTitle1,#mermaid-1558415763951 .sectionTitle2,#mermaid-1558415763951 .sectionTitle3{fill:#333}#mermaid-1558415763951 .sectionTitle{text-anchor:start;font-size:11px;text-height:14px}#mermaid-1558415763951 .grid .tick{stroke:#d3d3d3;opacity:.3;shape-rendering:crispEdges}#mermaid-1558415763951 .grid path{stroke-width:0}#mermaid-1558415763951 .today{fill:none;stroke:red;stroke-width:2px}#mermaid-1558415763951 .task{stroke-width:2}#mermaid-1558415763951 .taskText{text-anchor:middle;font-size:11px}#mermaid-1558415763951 .taskTextOutsideRight{fill:#000;text-anchor:start;font-size:11px}#mermaid-1558415763951 .taskTextOutsideLeft{fill:#000;text-anchor:end;font-size:11px}#mermaid-1558415763951 .taskText0,#mermaid-1558415763951 .taskText1,#mermaid-1558415763951 .taskText2,#mermaid-1558415763951 .taskText3{fill:#fff}#mermaid-1558415763951 .task0,#mermaid-1558415763951 .task1,#mermaid-1558415763951 .task2,#mermaid-1558415763951 .task3{fill:#8a90dd;stroke:#534fbc}#mermaid-1558415763951 .taskTextOutside0,#mermaid-1558415763951 .taskTextOutside1,#mermaid-1558415763951 .taskTextOutside2,#mermaid-1558415763951 .taskTextOutside3{fill:#000}#mermaid-1558415763951 .active0,#mermaid-1558415763951 .active1,#mermaid-1558415763951 .active2,#mermaid-1558415763951 .active3{fill:#bfc7ff;stroke:#534fbc}#mermaid-1558415763951 .activeText0,#mermaid-1558415763951 .activeText1,#mermaid-1558415763951 .activeText2,#mermaid-1558415763951 .activeText3{fill:#000!important}#mermaid-1558415763951 .done0,#mermaid-1558415763951 .done1,#mermaid-1558415763951 .done2,#mermaid-1558415763951 .done3{stroke:grey;fill:#d3d3d3;stroke-width:2}#mermaid-1558415763951 .doneText0,#mermaid-1558415763951 .doneText1,#mermaid-1558415763951 .doneText2,#mermaid-1558415763951 .doneText3{fill:#000!important}#mermaid-1558415763951 .crit0,#mermaid-1558415763951 .crit1,#mermaid-1558415763951 .crit2,#mermaid-1558415763951 .crit3{stroke:#f88;fill:red;stroke-width:2}#mermaid-1558415763951 .activeCrit0,#mermaid-1558415763951 .activeCrit1,#mermaid-1558415763951 .activeCrit2,#mermaid-1558415763951 .activeCrit3{stroke:#f88;fill:#bfc7ff;stroke-width:2}#mermaid-1558415763951 .doneCrit0,#mermaid-1558415763951 .doneCrit1,#mermaid-1558415763951 .doneCrit2,#mermaid-1558415763951 .doneCrit3{stroke:#f88;fill:#d3d3d3;stroke-width:2;cursor:pointer;shape-rendering:crispEdges}#mermaid-1558415763951 .activeCritText0,#mermaid-1558415763951 .activeCritText1,#mermaid-1558415763951 .activeCritText2,#mermaid-1558415763951 .activeCritText3,#mermaid-1558415763951 .doneCritText0,#mermaid-1558415763951 .doneCritText1,#mermaid-1558415763951 .doneCritText2,#mermaid-1558415763951 .doneCritText3{fill:#000!important}#mermaid-1558415763951 .titleText{text-anchor:middle;font-size:18px;fill:#000}#mermaid-1558415763951 g.classGroup text{fill:#9370db;stroke:none;font-family:trebuchet ms,verdana,arial;font-size:10px}#mermaid-1558415763951 g.classGroup rect{fill:#ececff;stroke:#9370db}#mermaid-1558415763951 g.classGroup line{stroke:#9370db;stroke-width:1}#mermaid-1558415763951 .classLabel .box{stroke:none;stroke-width:0;fill:#ececff;opacity:.5}#mermaid-1558415763951 .classLabel .label{fill:#9370db;font-size:10px}#mermaid-1558415763951 .relation{stroke:#9370db;stroke-width:1;fill:none}#mermaid-1558415763951 #compositionEnd,#mermaid-1558415763951 #compositionStart{fill:#9370db;stroke:#9370db;stroke-width:1}#mermaid-1558415763951 #aggregationEnd,#mermaid-1558415763951 #aggregationStart{fill:#ececff;stroke:#9370db;stroke-width:1}#mermaid-1558415763951 #dependencyEnd,#mermaid-1558415763951 #dependencyStart,#mermaid-1558415763951 #extensionEnd,#mermaid-1558415763951 #extensionStart{fill:#9370db;stroke:#9370db;stroke-width:1}#mermaid-1558415763951 .branch-label,#mermaid-1558415763951 .commit-id,#mermaid-1558415763951 .commit-msg{fill:#d3d3d3;color:#d3d3d3}#mermaid-1558415763951 {
color: rgb(0, 0, 0);
font: normal normal 400 normal 14px / 25.2px Verdana, Arial, Helvetica, sans-serif;
}
解析注解
生成 Mapper Model
将 Model 按规则写入 Freemarker 模板中, 并生成 Impl 文件
生成Impl对象, 转换时调用

具体解析

具体的解析逻辑是将解析注解内容转化为 Mapper model 对象,然后将 Mapper model 写入 freemarker 模板中。

处理框架

整个注解的解析是通过 java compile[1] 实现的,逻辑包含在MappingProcessor.process 函数中,并通过 MapperGenerationVisitor 进行解析。

	@Override
public boolean process(
final Set<? extends TypeElement> annotations,
final RoundEnvironment roundEnvironment) {
// 遍历需要处理的注解
for ( TypeElement oneAnnotation : annotations ) {
		<span class="hljs-comment">//Indicates that the annotation's type isn't on the class path of the compiled</span>
<span class="hljs-comment">//project. Let the compiler deal with that and print an appropriate error.</span>
<span class="hljs-keyword">if</span> ( oneAnnotation.getKind() != ElementKind.ANNOTATION_TYPE ) {
<span class="hljs-keyword">continue</span>;
}

// 遍历包含 Mapper 注解的 interface and class , 例如 org.mapstruct.ap.test.conversion.SourceTargetMapper

for ( Element oneAnnotatedElement : roundEnvironment.getElementsAnnotatedWith( oneAnnotation ) ) {

// MapperGenerationVisitor 解析每个Mapper 注解的内容 成为一个 Model

oneAnnotatedElement.accept( new MapperGenerationVisitor( processingEnv ), null );

}

}

	<span class="hljs-keyword">return</span> ANNOTATIONS_CLAIMED_EXCLUSIVELY;
}

解析逻辑

MapperGenerationVisitor 负责解析注解为 Mapper model, 并写入 ftl 模板文件中。

MapperGenerationVisitor.retrieveModel 包含了具体的解析逻辑,将注解内容转化为 Mapper Model。

ModelWriter 负责将 Mapper Model 写入 ftl 模板中。

整个逻辑都是围绕 Mapper model 展开的, Mapper 包含如下内容:

	private final String packageName; // 包的名称
<span class="hljs-keyword">private</span> <span class="hljs-keyword">final</span> String interfaceName; <span class="hljs-comment">// 接口名称</span>

<span class="hljs-keyword">private</span> <span class="hljs-keyword">final</span> String implementationName; <span class="hljs-comment">// 应用名称</span>

<span class="hljs-keyword">private</span> <span class="hljs-keyword">final</span> List&lt;BeanMapping&gt; beanMappings; <span class="hljs-comment">// 一系列的 mapping 信息, 每个 method 对应一个 BeanMapping</span>

每一个 BeanMapping 对应一个转换函数,它的格式如下:


private final Type sourceType; // 函数的输入参数类型
private final Type targetType; // 函数的结果参数类型
private final List<PropertyMapping> propertyMappings; // 转换函数的每个属性的信息
private final MappingMethod mappingMethod; // 映射的函数
private final MappingMethod reverseMappingMethod; // 翻转映射的函数
private final boolean isIterableMapping; // 是不是迭代

例如 SourceTargetMapper 接口:


@Mapper
public interface SourceTargetMapper {
<span class="hljs-keyword">public</span> <span class="hljs-keyword">static</span> SourceTargetMapper INSTANCE = Mappers.getMapper( SourceTargetMapper.class );

<span class="hljs-meta">@Mappings</span>({
<span class="hljs-meta">@Mapping</span>(source = <span class="hljs-string">"qax"</span>, target = <span class="hljs-string">"baz"</span>),
<span class="hljs-meta">@Mapping</span>(source = <span class="hljs-string">"baz"</span>, target = <span class="hljs-string">"qax"</span>)
})
<span class="hljs-function">Target <span class="hljs-title">sourceToTarget</span><span class="hljs-params">(Source source)</span></span>; <span class="hljs-function">Source <span class="hljs-title">targetToSource</span><span class="hljs-params">(Target target)</span></span>;

}

映射为 Mapper Model 为:

{
"beanMappings":[
{
"iterableMapping":false,
"mappingMethod":{
"name":"sourceToTarget",
"parameterName":"source"
},
"propertyMappings":[
{
"fromConversion":"target.getFoo().intValue()",
"sourceName":"foo",
"sourceType":{
"name":"int",
"primitive":true
},
"targetName":"foo",
"targetType":{
"name":"Long",
"packageName":"java.lang",
"primitive":false
},
"toConversion":"Long.valueOf( source.getFoo() )"
},
Object{...},
Object{...},
Object{...},
Object{...}
],
"reverseMappingMethod":{
"name":"targetToSource",
"parameterName":"target"
},
"sourceType":{
"name":"Source",
"packageName":"org.mapstruct.ap.test.conversion",
"primitive":false
},
"targetType":{
"name":"Target",
"packageName":"org.mapstruct.ap.test.conversion",
"primitive":false
}
}
],
"implementationName":"SourceTargetMapperImpl",
"interfaceName":"SourceTargetMapper",
"packageName":"org.mapstruct.ap.test.conversion"
}

写入模板

写入模板是使用 freemarker 进行编写的,最初写入逻辑很简单,直接使用 ModelWriter 进行写入。ftl 模板的部分内容如下:


package ${packageName}; import java.util.ArrayList;

import java.util.List; public class ${implementationName} implements ${interfaceName} {

上面的 ${packageName}对应的就是 Mapper Model 中的 packageName。

参考

  1. javax.lang.model.element.Element
  2. 编译器 API

原文地址:https://www.cnblogs.com/SpeakSoftlyLove/p/9794661.html

mapStruct笔记的更多相关文章

  1. Java实体映射工具MapStruct的使用

    官网地址:http://mapstruct.org/ MapStruct 是一个代码生成器,简化了不同的 Java Bean 之间映射的处理,所谓的映射指的就是从一个实体变化成一个实体.例如我们在实际 ...

  2. mapstruct 快速使用

    mapstruct 快速使用 mapstruct 主要的作用则是用来复制对象字段使用,功能非常的强大.在没有使用 mapstruct 之前可能都在使用 BeanUtils ,但是 BeanUtils ...

  3. git-简单流程(学习笔记)

    这是阅读廖雪峰的官方网站的笔记,用于自己以后回看 1.进入项目文件夹 初始化一个Git仓库,使用git init命令. 添加文件到Git仓库,分两步: 第一步,使用命令git add <file ...

  4. js学习笔记:webpack基础入门(一)

    之前听说过webpack,今天想正式的接触一下,先跟着webpack的官方用户指南走: 在这里有: 如何安装webpack 如何使用webpack 如何使用loader 如何使用webpack的开发者 ...

  5. SQL Server技术内幕笔记合集

    SQL Server技术内幕笔记合集 发这一篇文章主要是方便大家找到我的笔记入口,方便大家o(∩_∩)o Microsoft SQL Server 6.5 技术内幕 笔记http://www.cnbl ...

  6. PHP-自定义模板-学习笔记

    1.  开始 这几天,看了李炎恢老师的<PHP第二季度视频>中的“章节7:创建TPL自定义模板”,做一个学习笔记,通过绘制架构图.UML类图和思维导图,来对加深理解. 2.  整体架构图 ...

  7. PHP-会员登录与注册例子解析-学习笔记

    1.开始 最近开始学习李炎恢老师的<PHP第二季度视频>中的“章节5:使用OOP注册会员”,做一个学习笔记,通过绘制基本页面流程和UML类图,来对加深理解. 2.基本页面流程 3.通过UM ...

  8. NET Core-学习笔记(三)

    这里将要和大家分享的是学习总结第三篇:首先感慨一下这周跟随netcore官网学习是遇到的一些问题: a.官网的英文版教程使用的部分nuget包和我当时安装的最新包版本不一致,所以没法按照教材上给出的列 ...

  9. springMVC学习笔记--知识点总结1

    以下是学习springmvc框架时的笔记整理: 结果跳转方式 1.设置ModelAndView,根据view的名称,和视图渲染器跳转到指定的页面. 比如jsp的视图渲染器是如下配置的: <!-- ...

随机推荐

  1. android控件之webview和js与java交互

    首先添加权限:<uses-permission android:name="android.permission.INTERNET"/> 布局文件: <Relat ...

  2. 在colab上运行style-transfer

    1,  打开chrome浏览器,输入以下网址,打开风格转换主文件 https://colab.research.google.com/github/Hvass-Labs/TensorFlow-Tuto ...

  3. check_http.c:312: error: ‘ssl_version’

    安装nagios-plugins-1.4.16,安装的过程中出现了错误,提示如下.check_http.c:312: error: ‘ssl_version’ undeclared (first us ...

  4. 做OJ项目时遇到的坑

    1.js代码写在Dom加载前,导致highcharts在ie8能够显示,而ie高版本和其他浏览器不能显示 我的理解:由于IE8和其他浏览器的js解析机制不同,ie8是在等dom全部加载完才开始执行js ...

  5. 微软将于12月起开始推送Windows 10 Mobile

    [环球科技报道 记者 陈薇]据瘾科技网站10月8日消息,根据微软Lumia官方Faceboo发布的消息,新版系统Windows 10 Mobile 将会12月起陆续开始推送. 推送的具体时程根据地区. ...

  6. Oracle错误 1053: 该服务没有响应启动或控制请求

    在服务中,启动oracleDBConsolenewdb服务时,出现了 错误 1053: 该服务没有响应启动或控制请求 在网上查了以后你会发现这是一个非常宽泛的错误,然而我们的建议是去看传说中的orac ...

  7. HDU 5778 abs (暴力枚举)

    abs Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others) Problem De ...

  8. C#飞行棋总结

    以下是掷色子的一个代码,比较有代表性,里面的逻辑和内容都已注释,可通过注释了解这一方法的运作模式. public static void RowTouZi(int playerPos) //掷色子 { ...

  9. Python 使用random模块生成随机数

    需要先导入 random  模块,然后通过 random 静态对象调用该一些方法. random() 函数中常见的方法如下: # coding: utf-8 # Team : Quality Mana ...

  10. [整理] webpack+vuecli打包生成资源相对引用路径与背景图片的正确引用

    webpack+vuecli打包生成资源相对引用路径与背景图片的正确引用 https://www.cnblogs.com/moqiutao/p/7496718.html