前言

MyBatis是基于“数据库结构不可控”的思想建立的,也就是我们希望数据库遵循第三范式或BCNF,但实际事与愿违,那么结果集映射就是MyBatis为我们提供这种理想与现实间转换的手段了,而resultMap就是结果集映射的配置标签了。 
在深入ResultMap标签前,我们需要了解从SQL查询结果集到JavaBean或POJO实体的过程。

从SQL查询结果到领域模型实体 

  1. 通过JDBC查询得到ResultSet对象
  2. 遍历ResultSet对象并将每行数据暂存到HashMap实例中,以结果集的字段名或字段别名为键,以字段值为值
  3. 根据ResultMap标签的type属性通过反射实例化领域模型
  4. 根据ResultMap标签的type属性和id、result等标签信息将HashMap中的键值对,填充到领域模型实例中并返回

一、resultMap

1、属性说明

  • id属性 ,resultMap标签的标识。
  • type属性 ,返回值的全限定类名,或类型别名。
  • autoMapping属性 ,值范围true(默认值)|false, 设置是否启动自动映射功能,自动映射功能就是自动查找与字段名小写同名的属性名,并调用setter方法。而设置为false后,则需要在resultMap内明确注明映射关系才会调用对应的setter方法。

2、基本作用:建立SQL查询结果字段与实体属性的映射关系 
示例1:通过setter构造领域模型

  1. public class EStudent{
  2. private long id;
  3. private String name;
  4. private int age;
  5. // getter,setter方法
  6. /**
  7. * 必须提供一个无参数的构造函数
  8. */
  9. public EStudent(){}
  10. }
  1. <select id="getStudent" resultMap="getStudentRM">
  2. SELECT ID, Name, Age
  3. FROM TStudent
  4. </select>
  5. <resultMap id="getStudentRM" type="EStudnet">
  6. <id property="id" column="ID"/>
  7. <result property="studentName" column="Name"/>
  8. <result property="studentAge" column="Age"/>
  9. </resultMap>

子元素说明:

  • id元素 ,用于设置主键字段与领域模型属性的映射关系
  • result元素 ,用于设置普通字段与领域模型属性的映射关系

id、result语句属性配置细节:

属性 描述
property 需要映射到JavaBean 的属性名称。
column 数据表的列名或者标签别名。
javaType 一个完整的类名,或者是一个类型别名。如果你匹配的是一个JavaBean,那MyBatis 通常会自行检测到。然后,如果你是要映射到一个HashMap,那你需要指定javaType 要达到的目的。
jdbcType 数据表支持的类型列表。这个属性只在insert,update 或delete 的时候针对允许空的列有用。JDBC 需要这项,但MyBatis 不需要。如果你是直接针对JDBC 编码,且有允许空的列,而你要指定这项。
typeHandler 使用这个属性可以覆写类型处理器。这项值可以是一个完整的类名,也可以是一个类型别名。

示例2:通过构造函数构造领域模型

  1. <select id="getStudent" resultMap="getStudentRM">
  2. SELECT ID, Name, Age
  3. FROM TStudent
  4. </select>
  5. <resultMap id="getStudentRM" type="EStudnet">
  6. <constructor>
  7. <idArg column="ID" javaType="_long"/>
  8. <arg column="Name" javaType="String"/>
  9. <arg column="Age" javaType="_int"/>
  10. </constructor>
  11. </resultMap>

子元素说明:

  • constructor元素 ,指定使用指定参数列表的构造函数来实例化领域模型。注意:其子元素顺序必须与参数列表顺序对应
  • idArg子元素 ,标记该入参为主键
  • arg子元素 ,标记该入参为普通字段(主键使用该子元素设置也是可以的)

3、一对一关系、一对多关系查询

  1. 注意:在采用嵌套结果的方式查询一对一、一对多关系时,必须要通过resultMap下的idresult标签来显式设置属性/字段映射关系,否则在查询多条记录时会仅仅返回最后一条记录的情况。

association联合

联合元素用来处理“一对一”的关系。需要指定映射的Java实体类的属性,属性的javaType(通常MyBatis 自己会识别)。对应的数据库表的列名称。如果想覆写的话返回结果的值,需要指定typeHandler。 
不同情况需要告诉MyBatis 如何加载一个联合。MyBatis 可以用两种方式加载:

  • select: 执行一个其它映射的SQL 语句返回一个Java实体类型。较灵活;
  • resultsMap: 使用一个嵌套的结果映射来处理通过join查询结果集,映射成Java实体类型。

例如,一个班级对应一个班主任。 
首先定义好班级中的班主任 private TeacherEntity teacherEntity;

使用select实现联合 
例:班级实体类中有班主任的属性,通过联合在得到一个班级实体时,同时映射出班主任实体。 
这样可以直接复用在TeacherMapper.xml文件中定义好的查询teacher根据其ID的select语句。而且不需要修改写好的SQL语句,只需要直接修改resultMap即可。

ClassMapper.xml文件部分内容:

  1. <resultMap type="ClassEntity" id="classResultMap">
  2. <id property="classID" column="CLASS_ID" />
  3. <result property="className" column="CLASS_NAME" />
  4. <result property="classYear" column="CLASS_YEAR" />
  5. <association property="teacherEntity" column="TEACHER_ID" select="getTeacher"/>
  6. </resultMap>
  7. <select id="getClassByID" parameterType="String" resultMap="classResultMap">
  8. SELECT * FROM CLASS_TBL CT
  9. WHERE CT.CLASS_ID = #{classID};
  10. </select>

TeacherMapper.xml文件部分内容:

  1. <resultMap type="TeacherEntity" id="teacherResultMap">
  2. <id property="teacherID" column="TEACHER_ID" />
  3. <result property="teacherName" column="TEACHER_NAME" />
  4. <result property="teacherSex" column="TEACHER_SEX" />
  5. <result property="teacherBirthday" column="TEACHER_BIRTHDAY"/>
  6. <result property="workDate" column="WORK_DATE"/>
  7. <result property="professional" column="PROFESSIONAL"/>
  8. </resultMap>
  9. <select id="getTeacher" parameterType="String" resultMap="teacherResultMap">
  10. SELECT *
  11. FROM TEACHER_TBL TT
  12. WHERE TT.TEACHER_ID = #{teacherID}
  13. </select>

使用resultMap实现联合 
与上面同样的功能,查询班级,同时查询器班主任。需在association中添加resultMap(在teacher的xml文件中定义好的),新写sql(查询班级表left join教师表),不需要teacher的select。

修改ClassMapper.xml文件部分内容:

  1. <resultMap type="ClassEntity" id="classResultMap">
  2. <id property="classID" column="CLASS_ID" />
  3. <result property="className" column="CLASS_NAME" />
  4. <result property="classYear" column="CLASS_YEAR" />
  5. <association property="teacherEntity" column="TEACHER_ID" resultMap="teacherResultMap"/>
  6. </resultMap>
  7. <select id="getClassAndTeacher" parameterType="String" resultMap="classResultMap">
  8. SELECT *
  9. FROM CLASS_TBL CT LEFT JOIN TEACHER_TBL TT ON CT.TEACHER_ID = TT.TEACHER_ID
  10. WHERE CT.CLASS_ID = #{classID};
  11. </select>

其中的teacherResultMap请见上面TeacherMapper.xml文件部分内容中。

collection聚集

聚集元素用来处理“一对多”的关系。需要指定映射的Java实体类的属性,属性的javaType(一般为ArrayList);列表中对象的类型ofType(Java实体类);对应的数据库表的列名称; 
不同情况需要告诉MyBatis 如何加载一个聚集。MyBatis 可以用两种方式加载: 
1. select: 执行一个其它映射的SQL 语句返回一个Java实体类型。较灵活; 
2. resultsMap: 使用一个嵌套的结果映射来处理通过join查询结果集,映射成Java实体类型。

例如,一个班级有多个学生。 
首先定义班级中的学生列表属性:private List<StudentEntity> studentList;

使用select实现聚集 
用法和联合很类似,区别在于,这是一对多,所以一般映射过来的都是列表。所以这里需要定义javaType为ArrayList,还需要定义列表中对象的类型ofType,以及必须设置的select的语句名称(需要注意的是,这里的查询student的select语句条件必须是外键classID)。

ClassMapper.xml文件部分内容:

  1. <resultMap type="ClassEntity" id="classResultMap">
  2. <id property="classID" column="CLASS_ID" />
  3. <result property="className" column="CLASS_NAME" />
  4. <result property="classYear" column="CLASS_YEAR" />
  5. <association property="teacherEntity" column="TEACHER_ID" select="getTeacher"/>
  6. <collection property="studentList" column="CLASS_ID" javaType="ArrayList" ofType="StudentEntity" select="getStudentByClassID"/>
  7. </resultMap>
  8. <select id="getClassByID" parameterType="String" resultMap="classResultMap">
  9. SELECT * FROM CLASS_TBL CT
  10. WHERE CT.CLASS_ID = #{classID};
  11. </select>

StudentMapper.xml文件部分内容:

  1. <!-- java属性,数据库表字段之间的映射定义 -->
  2. <resultMap type="StudentEntity" id="studentResultMap">
  3. <id property="studentID" column="STUDENT_ID" />
  4. <result property="studentName" column="STUDENT_NAME" />
  5. <result property="studentSex" column="STUDENT_SEX" />
  6. <result property="studentBirthday" column="STUDENT_BIRTHDAY" />
  7. </resultMap>
  8. <!-- 查询学生list,根据班级id -->
  9. <select id="getStudentByClassID" parameterType="String" resultMap="studentResultMap">
  10. <include refid="selectStudentAll" />
  11. WHERE ST.CLASS_ID = #{classID}
  12. </select>

使用resultMap实现聚集 
使用resultMap,就需要重写一个sql,left join学生表。

  1. <resultMap type="ClassEntity" id="classResultMap">
  2. <id property="classID" column="CLASS_ID" />
  3. <result property="className" column="CLASS_NAME" />
  4. <result property="classYear" column="CLASS_YEAR" />
  5. <association property="teacherEntity" column="TEACHER_ID" resultMap="teacherResultMap"/>
  6. <collection property="studentList" column="CLASS_ID" javaType="ArrayList" ofType="StudentEntity" resultMap="studentResultMap"/>
  7. </resultMap>
  8. <select id="getClassAndTeacherStudent" parameterType="String" resultMap="classResultMap">
  9. SELECT *
  10. FROM CLASS_TBL CT
  11. LEFT JOIN STUDENT_TBL ST
  12. ON CT.CLASS_ID = ST.CLASS_ID
  13. LEFT JOIN TEACHER_TBL TT
  14. ON CT.TEACHER_ID = TT.TEACHER_ID
  15. WHERE CT.CLASS_ID = #{classID};
  16. </select>

其中的teacherResultMap请见上面TeacherMapper.xml文件部分内容中。studentResultMap请见上面StudentMapper.xml文件部分内容中。

4. 动态映射关系 
通过 discriminator子元素 (鉴别器)可以实现动态映射关系信息的设置。具体示例如下:

  1. public class EStudent{
  2. private long id;
  3. private String name;
  4. private String juniorHighSchool;
  5. private String seniorHighSchool;
  6. private int during; // 在本校就读时间
  7. // getter,setter方法
  8. /**
  9. * 必须提供一个无参数的构造函数
  10. */
  11. public EStudent(){}
  12. }

情景:查询学生信息的seniorHighSchool信息,若就读时间during字段值为4、5、6时,则以juniorHighSchool字段作所为seniorHighSchool信息。

  1. <select id="getStundent" resultMap="rm">
  2. SELECT ID, Name, JuniorHighSchool, SeniorHighSchool, during
  3. FROM TStudent
  4. </select>
  5. <resultMap id="rm" type="EStudent">
  6. // 若不加这句,则当将juniorHighSchool赋予给seniorHighSchool属性时,juniorHighSchool属性将为null
  7. <result column="juniorHighSchool" property="juniorHighSchool"/>
  8. <discriminator column="during" javaType="_int">
  9. // 形式1:通过resultType设置动态映射信息
  10. <case value="4" resultType="EStudent">
  11. <result column="juniorHighSchool" property="seniorHighSchool"/>
  12. </case>
  13. // 形式2: 通过resultMap设置动态映射信息
  14. <case value="5" resultMap="dynamicRM"/>
  15. <case value="6" resultMap="dynamicRM"/>
  16. </discriminator>
  17. </resultMap>
  18. <resultMap id="dynamicRM" type="EStudent">
  19. <result column="juniorHighSchool" property="seniorHighSchool"/>
  20. </resultMap>

注意:上面关于 discriminator子元素 的 case元素 的 resultType属性 和 resultMap元素 的 type属性 ,均不是直指返回的领域模型类型,而是指定根据判断条件后得到映射关系,可通过 id子元素 和 result子元素 重写映射关系。

5. id元素,result元素,idArg元素,arg元素,discriminator元素的共同属性

  • javaType属性 :Java类的全限定名,或别名
  • jdbcType属性 :JDBC类型, JDBC类型为CUD操作时列可能为空时进行处理
  • typeHandler属性 :指定类型处理器的全限定类名或类型别名
  • column属性 :指定SQL查询结果的字段名或字段别名。将用于JDBC的 resultSet.getString(columnName)

mybatis属性详解的更多相关文章

  1. 一、Mybatis配置详解

    Mybatis配置详解 XML配置文件层次结构 下图展示了mybatis-config.xml的全部配置元素 properties元素 properties是一个配置属性的元素,让我们能在配置文件的上 ...

  2. ORM框架对比以及Mybatis配置文件详解

    ORM框架对比以及Mybatis配置文件详解 0.数据库操作框架的历程 (1) JDBC ​ JDBC(Java Data Base Connection,java数据库连接)是一种用于执行SQL语句 ...

  3. android:exported 属性详解

    属性详解 标签: android 2015-06-11 17:47 27940人阅读 评论(7) 收藏 举报 分类: Android(95) 项目点滴(25) 昨天在用360扫描应用漏洞时,扫描结果, ...

  4. OutputCache属性详解(一)一Duration、VaryByParam

    目录 OutputCache概念学习 OutputCache属性详解(一) OutputCache属性详解(二) OutputCache属性详解(三) OutputCache属性详解(四)— SqlD ...

  5. OutputCache属性详解(二)一 Location

    目录 OutputCache概念学习 OutputCache属性详解(一) OutputCache属性详解(二) OutputCache属性详解(三) OutputCache属性详解(四)— SqlD ...

  6. OutputCache属性详解(三)— VaryByHeader,VaryByCustom

    目录 OutputCache概念学习 OutputCache属性详解(一) OutputCache属性详解(二) OutputCache属性详解(三) OutputCache属性详解(四)— SqlD ...

  7. OutputCache属性详解(四)— SqlDependency

    目录 OutputCache概念学习 OutputCache属性详解(一) OutputCache属性详解(二) OutputCache属性详解(三) OutputCache属性详解(四)— SqlD ...

  8. WPF依赖属性详解

    WPF依赖属性详解 WPF 依赖属性 英文译为 Dependency Properties,是WPF引入的一种新类型的属性,在WPF中有着极为广泛的应用,在WPF中对于WPF Dependency P ...

  9. HTML video 视频标签全属性详解

    HTML 5 video 视频标签全属性详解   现在如果要在页面中使用video标签,需要考虑三种情况,支持Ogg Theora或者VP8(如果这玩意儿没出事的话)的(Opera.Mozilla.C ...

随机推荐

  1. u3d开发中可能会遇到的设计模式

    最近一段时间,面试了一些程序员,当然主要招聘的岗位是Unity3D开发.面试过程中对于三年以上的程序员我都会问其在开发中是否会总结一些常用的设计模式和设计方法,当然目的只是想了解程序员的自我学习情况以 ...

  2. [].slice.call(arguments,1)

    [转] 前言 今天偶然翻资料看到一个叫做软绑定的函数,用来确定this的; 原代码 if(!Function.prototype.softBind){ Function.prototype.softB ...

  3. 关于React中状态保存的研究

    在使用react搭配react-router做应用的时候,你可能遇到这样的问题,当我从第一个页面过渡到第二个页面,然后返回之后,发现之前的页面的状态全部不见了,即回到了初始的状态. 这点在页面存在多个 ...

  4. 透过一道面试题来探探JavaScript中执行上下文和变量对象的底

    在做面试题之前,我们先搞清楚两个概念 执行上下文(execution context) 变量对象(variable object) 执行上下文 我们都知道JavaScript的作用域一共分三种 全局作 ...

  5. Yii2.0源码阅读-视图(View)渲染过程

    之前的文章我们根据源码的分析,弄清了Yii如何处理一次请求,以及根据解析的路由如何调用控制器中的action,那接下来好奇的可能就是,我在控制器action中执行了return $this->r ...

  6. 前端Web开发MVC模式-入门示例

    MVC概论起初来之桌面应用开发.其实java的structs框架最能体现MVC框架:model模型是理解成服务器端的模块程序:view为发送给客服端的内容:control为servlet程序控制跳转和 ...

  7. js面向对象学习笔记(四):对象的混合写法

    //对象的混合写法//1.构造函数function 构造函数() { this.属性}构造函数.原型.方法 = function () {};//调用var 对象1 = new 构造函数();对象1. ...

  8. 查找第k小的元素(O(n)递归解法)

    今天分享一个小技巧,虽然是小技巧但是还是很有价值的,曾经是微软的面试题.题目是这样的,一个无序的数组让你找出第k小的元素,我当时看到这道题的时候也像很多人一样都是按普通的思维,先排序在去第K个,但是当 ...

  9. C++课程设计类作业2

    不要问我一个晚上在干啥,就写写这种烦到极点的类,啰嗦! #include <bits/stdc++.h> using namespace std; class complexed { pu ...

  10. 计蒜客:百度的科学计算器(简单)【python神解】

    题目链接:https://nanti.jisuanke.com/t/15504 题解:python大法好啊,三行代码无人能敌啊! 下面给出AC代码: b=input() a=input() print ...