经常会用到mybatis的Collection标签来做级联查询或递归查询,现通过一个伪例来简单的说明一下使用中的关键点:

首先先列出三个表,给出一个场景:

1,角色表 t_role( id,name )  
2,菜单表 t_menu( id, name, pid ) 菜单表是个有个pid,指向上级菜单的id,所以是个树形表
3,角色菜单关联表 t_role_menu( role_id, menu_id ) 角色和菜单对多关联
java对象如下:

//菜单对象
@Set
@Get
public class Menu {
private Long id;
private String name;
private Long pid;
private List<Menu> subMenus;//子菜单集合
} //角色对象
@Set
@Get
public class Role {
private Long id;
private String name;
private List<Menu> menus;//关联菜单集合
}

一,简单关联查询:根据角色id查询角色,以及角色所关联的菜单的集合;

mapper xml文件配置如下:

<!--1,根据主键查询角色,及其关联的菜单-->
<select id="selectRoleById" parameterType="Long" resultMap="RoleResultMap">
select * from t_role where id = #{id}
</select> <!--2,角色结果映射-->
<resultMap id="RoleResultMap" type="Role">
<id column="id" jdbcType="BIGINT" property="id" />
<result column="name" jdbcType="VARCHAR" property="name" />
<collection property="menus" column="id" ofType="Menu" select="selectMenuByRoleId"/>
</resultMap> <!--3,根据角色id,查询菜单集合-->
<select id="selectMenuByRoleId" parameterType="Long" resultMap="MenuResultMap">
select a.* from t_menu a
join t_role_menu b on a.id = b.menu_id and b.role_id=#{roleId}<!--roleId这个变量名称可以随便写-->
</select> <!--4,菜单结果映射-->
<resultMap id="MenuResultMap" type="Menu">
<id column="id" jdbcType="BIGINT" property="id" />
<result column="name" jdbcType="VARCHAR" property="name" />
<result column="pid" jdbcType="BIGINT" property="name" />
</resultMap>

指定的角色以及关联的菜单集合,调用顺序已经按照序号1234排列,其中需要注意的有两点:

1,第2步,collection标签中的column=“id”,id是role查询结果映射中的column,而不是property;

2,第3步,参数#{roleId},这个名称可以随便起;

通过上边的例子,可以查询出角色关联的菜单,但是每个菜单对象的子菜单集合都是null,因为没有查询菜单的子菜单,下面引入下面的场景:递推查询。

二,递归查询(特殊的关联查询):根据菜单id查询菜单树,自己调用自己

mapper xml文件配置如下:

<!--1,根据主键查询菜单-->
<select id="selectMenuTreeById" parameterType="Role" resultMap="MenuResultMap">
select * from t_menu where id = #{id}
</select> <!--2,菜单结果映射-->
<resultMap id="MenuResultMap" type="Menu">
<id column="id" jdbcType="BIGINT" property="id" />
<result column="name" jdbcType="VARCHAR" property="name" />
<result column="pid" jdbcType="BIGINT" property="name" />
<collection property="subMenus" column="id" ofType="Menu" select="selectSubMenuByPid"/>
</resultMap> <!--3,查询id查询菜单-->
<select id="selectSubMenuByPid" parameterType="Role" resultMap="MenuResultMap">
select * from t_menu where pid = #{pid}
</select>

TreeById”方法就完成菜单树的调用,相比上边的关联查询少了一步结果映射,这是因为父查询和子查询的映射结果相同。注意的点也在于此。

三,多字段关联:把上边的递归查询稍微变一下,不通过菜单的id查询菜单树,而是查询指定角色的菜单树,应该怎么改一下呢?

我们把第一步的方法,改一下,参数变成角色id,并且通过关联表进行关联:

<!--1,根据角色id查询菜单-->

<select id="selectMenuByRoleId" parameterType="Long" resultMap="MenuResultMap">
select a.* from t_menu a
join t_role_menu b on a.id = b.menu_id and b.role_id=#{roleId}
</select>

这样是不是就可以了?答案是:NO!

因为这样只能保证根菜单是和角色关联的,下边的子菜单并不能保证与角色关联,对应的地推查询也需要增加关联,所以就引出了“多字段关联”的情况,完整的mapper xml文件配置如下:

<!--1,根据角色id查询菜单-->
<select id="selectMenuByRoleId" parameterType="Long" resultMap="MenuResultMap">
select a.*, b.role_id
from t_menu a
join t_role_menu b on a.id = b.menu_id and b.role_id=#{roleId}
</select> <!--2,菜单结果映射-->
<resultMap id="MenuResultMap" type="Menu">
<id column="id" jdbcType="BIGINT" property="id" />
<result column="name" jdbcType="VARCHAR" property="name" />
<result column="pid" jdbcType="BIGINT" property="name" />
<!--关联参数,与java实体没有关系-->
<result column="role_id" jdbcType="BIGINT" />
<collection property="subMenus" column="{pid=id,roleId=role_id}" ofType="Menu" select="selectSubMenuByPid"/>
</resultMap> <!--3,查询id查询菜单-->
<select id="selectSubMenuByPid" parameterType="java.util.Map" resultMap="MenuResultMap">
select a.*, b.role_id
from t_menu a
join t_role_menu b on a.id = b.menu_id and b.role_id=#{roleId}
where a.pid = #{pid}
</select>

1,需要在第一步的时候把关联的角色id(role_id)查出来,放到结果映射中,如第2步所示,增加了一个result标签,column就是role_id,但是不要加property(如果实际需要的话也可以加,但是需要在java对象中增加该属性),它不映射到java对象上,只是作为关联字段。

2,第2步,在collection标签中需要增加关联字段,多个column的写法如上所示,k=v的形式,k就是参数名,v就是引用的结果字段(column而不是property),同时在子查询中的参数类型需要改成Map,并且参数名不能随便写了,需要根据collection中的指定的参数名来使用。

MyBatis Collection小记—— 关联查询、递归查询、多字段关联的更多相关文章

  1. MyBatis关联查询,一对多关联查询

    实体关系图,一个国家对应多个城市 一对多关联查询可用三种方式实现: 单步查询,利用collection标签为级联属性赋值: 分步查询: 利用association标签进行分步查询: 利用collect ...

  2. MyBatis学习总结(三)——多表关联查询与动态SQL

    在上一章中我们学习了<MyBatis学习总结(二)——MyBatis核心配置文件与输入输出映射>,这一章主要是介绍一对一关联查询.一对多关联查询与动态SQL等内容. 一.多表关联查询 表与 ...

  3. MyBatis入门(二)—— 输入映射和输出映射、动态sql、关联查询

    一.输入映射和输出映射 1. parameterType(输入类型) 1.1 传递简单类型 <select id="getUserById" parameterType=&q ...

  4. MyBatis关联查询、多条件查询

    MyBatis关联查询.多条件查询 1.一对一查询 任务需求; 根据班级的信息查询出教师的相关信息 1.数据库表的设计 班级表: 教师表: 2.实体类的设计 班级表: public class Cla ...

  5. MyBatis从入门到放弃四:一对多关联查询

    前言 上篇学习了一对一关联查询,这篇我们学习一对多关联查询.一对多关联查询关键点则依然是配置resultMap,在resultMap中配置collection属性,别忽略了ofType属性. 搭建开发 ...

  6. MyBatis:一对多关联查询

    MyBatis从入门到放弃四:一对多关联查询 前言 上篇学习了一对一关联查询,这篇我们学习一对多关联查询.一对多关联查询关键点则依然是配置resultMap,在resultMap中配置collecti ...

  7. MyBatis关联查询,一对一关联查询

    数据库E-R关系 实体类 public class City { Long id; String name; Long countryId; Date lastUpdate; } public cla ...

  8. 关联查询 join on 和比较运算符 in

    join on多表之间的关联查询 写法select 字段 from 表1 t join 表2 s on t.字段1 = s.字段1 where 条件: 也可以这么写select 字段 from 表1 ...

  9. Ibatis组合键关联查询

    在使用Ibatis时,会经常遇到关联查询,一般有两种解决方案: 使用代码进行关联查询 使用Ibatis配置文件进行关联查询 使用代码进行关联查询不作解释,本次主要是针对Ibatis配置文件进行关联查询 ...

随机推荐

  1. HDC2021:HMS Core分析服务,数智化营销闭环方案帮助开发者实现精益增长

    10.22-10.24华为开发者大会2021(Together)在东莞如期举行.本次大会上,HMS Core华为分析服务作为多平台.跨设备的一站式数据分析平台以数据驱动业务智能决策为理念,带来了数智化 ...

  2. 【数据结构&算法】08-栈概念&源码

    目录 前言 栈的定义 定义 常见应用 栈的常见应用 进栈出栈变化形式 栈的抽象数据类型 栈的顺序存储结构及实现 栈的顺序存储结构 顺序栈 顺序栈的结构定义 两栈共享空间 栈的链式存储结构及实现 栈的链 ...

  3. 分享一下Eclipse中节省时间的技巧吧

    [初级技巧] ★★ 鼠标放在一个类名上面,会显示Javadoc.也可以通过屏幕下方的Javadoc面板来查看(你可以把它看成是MSDN的Java版). ★ 每个函数的第一行,左边有个圆圈,单击这个圆圈 ...

  4. [atAGC054D]ox

    对于两个字符串$s$和$t$(保证其中每一种字符个数相同),定义$s$和$t$的相对逆序对数为$s$得到$t$的最少交换次数,显然同种字符相对顺序保持不变,因此即依次编号后的逆序对数 问题不妨看作构造 ...

  5. [ARC117D]Miracle Tree

    将$E_{i}$从小到大排序(显然不会相同),假设$E_{p_{i}}$为从小到大第$i$小 此时,必然有$E_{p_{1}}=1$,否则可以将$E_{p_{i}}$都减去$E_{p_{1}}-1$, ...

  6. Spring Cloud Gateway限流实战

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...

  7. IPv4 寻址方式简介

    IPv4 支持三种不同类型的寻址模式.单播寻址方式.广播寻址方式和组播寻址方式.本章节我们来介绍这些寻址方式. 单播寻址方式 在这种模式下,数据只发送到一个目标主机.Destination Addre ...

  8. 7.2 k8s 基于PV、PVC搭建zookeeper 3节点集群

    1.PV,PVC介绍 1.1.StorageClass & PV & PVC关系图 Volumes 是最基础的存储抽象,其支持多种类型,包括本地存储.NFS.FC以及众多的云存储,我们 ...

  9. BZOJ 3729 - Gty的游戏(Staircase 博弈+时间轴分块)

    题面传送门 介于自己以前既没有写过 Staircase-Nim 的题解,也没写过时间轴分块的题解,所以现在就来写一篇吧(fog 首先考虑最极端的情况,如果图是一条链,并且链的一个端点是 \(1\),那 ...

  10. Codeforces 1089I - Interval-Free Permutations(析合树计数)

    Codeforces 题面传送门 & 洛谷题面传送门 首先题目中涉及排列的 interval,因此可以想到析合树.由于本蒟蒻太菜了以至于没有听过这种神仙黑科技,因此简单介绍一下这种数据结构:我 ...