最近在读刘增辉老师所著的《MyBatis从入门到精通》一书,很有收获,于是将自己学习的过程以博客形式输出,如有错误,欢迎指正,如帮助到你,不胜荣幸!

本篇博客主要讲解MyBatis中实现查询结果一对一映射的3种方式:

  1. 使用别名实现自动映射
  2. 使用resultMap配置
  3. 使用resultMap的association标签配置

1. 使用别名实现自动映射

假设有这样1个需求:根据用户id查询用户信息的同时获取用户拥有的角色,为了举例,我们假设一个用户只能拥有一个角色(实际情况肯定不是这样的)。

一般情况下,不建议直接修改数据库表对应的实体类,所以这里我们新建类SysUserExtend,让它继承自SysUser类,并添加SysRole类型的字段,如下所示:

package com.zwwhnly.mybatisaction.model;import java.util.List;public class SysUserExtend extends SysUser {/*** 用户角色*/private SysRole sysRole;public SysRole getSysRole() {return sysRole;}public void setSysRole(SysRole sysRole) {this.sysRole = sysRole;}
}

然后,我们在接口SysUserMapper中添加如下方法:

/*** 根据用户id获取用户信息和用户的角色信息** @param id* @return*/
SysUserExtend selectUserAndRoleById(Long id);

接着,在对应的SysUserMapper.xml中添加如下代码:

<select id="selectUserAndRoleById" resultType="com.zwwhnly.mybatisaction.model.SysUserExtend">SELECT u.id,u.user_name userName,u.user_password userPassword,u.user_email userEmail,u.create_time createTime,r.id "sysRole.id",r.role_name   "sysRole.roleName",r.enabled "sysRole.enabled",r.create_by   "sysRole.createBy",r.create_time "sysRole.createTime"FROM sys_user uINNER JOIN sys_user_role ur ON u.id = ur.user_idINNER JOIN sys_role r ON ur.role_id = r.idWHERE u.id = #{id}
</select>

这里要注意两点,第一点是这里的resultType,要设置成刚刚新建的SysUserExtend类,第二点是列的别名要设置成"sysRole.id"这样的格式,其中sysRole要和SysUserExtend类中的字段名保持一致,id要和SysRole类中的字段名保持一致。

最后,在SysUserMapperTest测试类中添加如下测试方法:

@Test
public void testSelectUserAndRoleById() {SqlSession sqlSession = getSqlSession();try {SysUserMapper sysUserMapper = sqlSession.getMapper(SysUserMapper.class);// 注意这里使用1001这个用户,因为这个用户只有1个角色SysUserExtend sysUserExtend = sysUserMapper.selectUserAndRoleById(1001L);Assert.assertNotNull(sysUserExtend);Assert.assertNotNull(sysUserExtend.getSysRole());} finally {sqlSession.close();}
}

运行测试代码,测试通过,输出日志如下:

DEBUG [main] - ==> Preparing: SELECT u.id, u.user_name userName, u.user_password userPassword, u.user_email userEmail, u.create_time createTime, r.id "sysRole.id", r.role_name "sysRole.roleName", r.enabled "sysRole.enabled", r.create_by "sysRole.createBy", r.create_time "sysRole.createTime" FROM sys_user u INNER JOIN sys_user_role ur ON u.id = ur.user_id INNER JOIN sys_role r ON ur.role_id = r.id WHERE u.id = ?

DEBUG [main] - ==> Parameters: 1001(Long)

TRACE [main] - <== Columns: id, userName, userPassword, userEmail, createTime, sysRole.id, sysRole.roleName, sysRole.enabled, sysRole.createBy, sysRole.createTime

TRACE [main] - <== Row: 1001, test, 123456, test@mybatis.tk, 2019-06-27 18:21:07.0, 2, 普通用户, 1, 1, 2019-06-27 18:21:12.0

DEBUG [main] - <== Total: 1

2. 使用resultMap配置

延用上面的需求,不过换成resultMap来配置映射。

首先,我们在接口SysUserMapper中添加如下方法:

/*** 根据用户id获取用户信息和用户的角色信息** @param id* @return*/
SysUserExtend selectUserAndRoleByIdResultMap(Long id);

然后在对应的SysUserMapper.xml中先添加如下resultMap:

<resultMap id="userRoleMap" type="com.zwwhnly.mybatisaction.model.SysUserExtend"><id property="id" column="id"/><result property="userName" column="user_name"/><result property="userPassword" column="user_password"/><result property="userEmail" column="user_email"/><result property="userInfo" column="user_info"/><result property="headImg" column="head_img" jdbcType="BLOB"/><result property="createTime" column="create_time" jdbcType="TIMESTAMP"/><!--role相关属性--><result property="sysRole.id" column="role_id"/><result property="sysRole.roleName" column="role_role_name"/><result property="sysRole.enabled" column="role_enabled"/><result property="sysRole.createBy" column="role_create_by"/><result property="sysRole.createTime" column="role_create_time" jdbcType="TIMESTAMP"/>
</resultMap>

为了避免多张表有相同的列名,所以这里配置role相关属性时,统一加上了"role_"这样的前缀,这里要和下方的查询语句中设置的列的别名保持一致:

<select id="selectUserAndRoleByIdResultMap" resultMap="userRoleMap">SELECT  u.id,u.user_name,u.user_password,u.user_email,u.create_time,r.id role_id,r.role_name role_role_name,r.enabled role_enabled,r.create_by role_create_by,r.create_time role_create_timeFROM sys_user uINNER JOIN sys_user_role ur ON u.id = ur.user_idINNER JOIN sys_role r ON ur.role_id = r.idWHERE u.id = #{id}
</select>

注意事项:这里的查询语句,我们使用的是resultMap而不是resultType。

由于测试代码和输出日志都和1中的几乎一致,所以这里不再列出。

可能有的同学还记得我们之前在SysUserMapper.xml中配置过这样的一个resultMap:

<resultMap id="sysUserMap" type="com.zwwhnly.mybatisaction.model.SysUser"><id property="id" column="id"/><result property="userName" column="user_name"/><result property="userPassword" column="user_password"/><result property="userEmail" column="user_email"/><result property="userInfo" column="user_info"/><result property="headImg" column="head_img" jdbcType="BLOB"/><result property="createTime" column="create_time" jdbcType="TIMESTAMP"/>
</resultMap>

然后就会想,我们刚刚定义的userRoleMap,前半部分和这个配置都是一样的,能复用这个sysUserMap的配置吗?

答案当然是可以,MyBatis是支持resultMap映射继承的。

所以,userRoleMap可以继承自sysUserMap,省略掉重复的配置,优化后的配置如下所示:

<resultMap id="userRoleMap" type="com.zwwhnly.mybatisaction.model.SysUserExtend" extends="sysUserMap"><!--role相关属性--><result property="sysRole.id" column="role_id"/><result property="sysRole.roleName" column="role_role_name"/><result property="sysRole.enabled" column="role_enabled"/><result property="sysRole.createBy" column="role_create_by"/><result property="sysRole.createTime" column="role_create_time" jdbcType="TIMESTAMP"/>
</resultMap>

3. 使用resultMap的association标签配置

上面的配置也可以用association标签来配置,配置方式如下所示(实现的效果是一样的):

<resultMap id="userRoleMap" type="com.zwwhnly.mybatisaction.model.SysUserExtend" extends="sysUserMap"><association property="sysRole" columnPrefix="role_" javaType="com.zwwhnly.mybatisaction.model.SysRole"><result property="id" column="id"/><result property="roleName" column="role_name"/><result property="enabled" column="enabled"/><result property="createBy" column="create_by"/><result property="createTime" column="create_time" jdbcType="TIMESTAMP"/></association>
</resultMap>

有想法的同学可能会觉得,角色表的映射配置完全可以独立出来配置啊,比如这样:

<resultMap id="roleMap" type="com.zwwhnly.mybatisaction.model.SysRole"><id property="id" column="id"/><result property="roleName" column="role_name"/><result property="enabled" column="enabled"/><result property="createBy" column="create_by"/><result property="createTime" column="create_time" jdbcType="TIMESTAMP"/>
</resultMap>

然后强大的MyBatis可以让association标签来直接使用roleMap,所以上面的association标签配置可以优化为:

<resultMap id="userRoleMap" type="com.zwwhnly.mybatisaction.model.SysUserExtend" extends="sysUserMap"><association property="sysRole" columnPrefix="role_"resultMap="roleMap"/>
</resultMap>

此时,roleMap我们是建在SysUserMapper.xml中的,不过实际使用中,这个roleMap放到SysRoleMapper.xml中会更合理一些。

不过要注意的是,将roleMap移动到SysRoleMapper.xml中后,引用roleMap时就要使用全名,如下所示:

<association property="sysRole" columnPrefix="role_"resultMap="com.zwwhnly.mybatisaction.mapper.SysRoleMapper.roleMap"/>

否则会报如下错误。

4. 源码及参考

源码地址:https://github.com/zwwhnly/mybatis-action.git,欢迎下载。

刘增辉《MyBatis从入门到精通》

5. 最后

打个小广告,欢迎扫码关注微信公众号:「申城异乡人」,定期分享Java技术干货,让我们一起进步。

转载于:https://www.cnblogs.com/zwwhnly/p/11169811.html

MyBatis从入门到精通(九):MyBatis高级结果映射之一对一映射相关推荐

  1. MyBatis从入门到精通(一)—MyBatis基础知识和快速入门

    Mybatis简介 原始jdbc操作(查询数据) Connection connection = null;PreparedStatement preparedStatement = null;Res ...

  2. MyBatis从入门到精通(1):MyBatis入门

    作为一个自学Java的自动化专业211大学本科生,在学习和实践过程中"趟了不少雷",所以有志于建立一个适合同样有热情学习Java技术的参考"排雷手册". 最近在 ...

  3. MyBatis从入门到精通(一):MyBatis入门

    最近在读刘增辉老师所著的<MyBatis从入门到精通>一书,很有收获,于是将自己学习的过程以博客形式输出,如有错误,欢迎指正,如帮助到你,不胜荣幸! 1. MyBatis简介 ​ 2001 ...

  4. MyBatis从入门到精通 PDF 完整版

    给大家带来的一篇关于MyBatis相关的电子书资源,介绍了关于MyBatis.入门到精通方面的内容,本书是由电子工业出版社出版,格式为PDF,资源大小116.8 MB,刘增辉编写,目前豆瓣.亚马逊.当 ...

  5. 视频教程-mybatis快速入门到精通-Java

    mybatis快速入门到精通 十年项目开发经验,主要从事java相关的开发,熟悉各种mvc开发框架. 王振伟 ¥18.00 立即订阅 扫码下载「CSDN程序员学院APP」,1000+技术好课免费看 A ...

  6. Mybatis从入门到精通下篇

    Mybatis从入门到精通下篇: 输入类型: 输出类型: ResultMap: 动态sql: if标签: where标签: sql片段: foreach标签: 关联查询: 以订单作为主体: 一对一查询 ...

  7. Mybatis从入门到精通(全)

    目录 前言 1. idea软件小技巧 2. mybatis优势 3. mybatis入门项目 3.1 项目具体步骤 3.2 创建数据表 3.3 创建maven项目 3.4 pom.xml配置文件 3. ...

  8. Mybatis从入门到精通读书笔记

    Mybatis从入门到精通 resultMap resultMap简介 resultMap resultMap简介 P25-P26

  9. Mybatis从入门到精通上篇

    Mybatis从入门到精通上篇: 学习过的持久层框架:DBUtils , Hibernate Mybatis就是类似于hibernate的orm持久层框架. Mybatis介绍: Mybatis是面向 ...

最新文章

  1. 从硬件到软件 统一沟通将引领通讯市场
  2. laravel批量插入报错:1292: Incorrect datetime value: '0000-00-00 00:00:00' for column 'TERM_DATE'
  3. Redis数据持久化
  4. Jacobian矩阵和Hessian矩阵的理解
  5. WINCE的FFMPEG交叉编译环境搭建
  6. go-zero:微服务框架
  7. Winform开发之ADO.NET对象Connection、Command、DataReader、DataAdapter、DataSet和DataTable简介...
  8. iOS:苹果内购实践
  9. 【WEB基础】HTML CSS 基础入门(2)选取工具:VS2019安装使用
  10. 软件工程需求分析文档模板
  11. 科学计算机计算内插法,线性插值计算器
  12. Idm在B站没有显示下载按钮
  13. 基于 HLS 创建 Golang 视频流服务器
  14. win10安装—手记
  15. java lint_Android静态代码检查-Lint
  16. ifft变换用java_利用FFT 及 IFFT实现傅立叶正反变换 | 学步园
  17. win10 64位注册TeeChart8.ocx
  18. 攻防世界-warmup详解
  19. 数字视频压缩的可行性
  20. dnf剑魂buff等级上限_DNF:半年前没人看得上,如今被开发成T0,这神话终于翻身了...

热门文章

  1. 人脸识别评价参数之FAR和FRR
  2. 小红书如何引流兼职粉?快点击打开看看吧
  3. C语言中如何用循环编译菱形,c语言for循环如何打印菱形
  4. Neuroimaging 海外职位信息
  5. 阿里安筱鹏:一文讲透数字化转型的本质!
  6. 太恐怖!黑客能把你的耳机变成麦克风用来监听
  7. 无线运动耳机品牌排行榜前十名,目前最火爆的六款运动耳机推荐
  8. 字体与版式设计 字体设计让你的信息变独特
  9. 猪猪的机器学习(十九)卷积神经网络
  10. 从月薪3千到3万,需要努力多久?