MyBatis教程:使用collection标签实现嵌套查询的方法

2019-07-1720:33:39后端程序开发Comments2,261 views字数 9546阅读模式

使用collection标签实现嵌套查询的方法。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

1. 需求升级

在上篇博客中,我们实现了需求:根据用户id查询用户信息的同时获取用户拥有的角色。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

因为角色可以拥有多个权限,所以本篇博客我们升级需求为:根据用户id查询用户信息的同时获取用户拥有的角色以及角色包含的权限。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

2. 实现方式

因为我们需要使用到权限表的映射,所以我们需要先在SysPrivilegeMapper.xml中添加如下映射:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

<resultMap id="sysPrivilegeMap" type="com.zwwhnly.mybatisaction.model.SysPrivilege">
    <id property="id" column="id"/>
    <result property="privilegeName" column="privilege_name"/>
    <result property="privilegeUrl" column="privilege_url"/>
</resultMap>

一般情况下不建议修改数据库表对应的实体类,所以这里我们新建类SysRoleExtend,让它继承SysRole类,并添加如下字段:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

package com.zwwhnly.mybatisaction.model;

import java.util.List;

public class SysRoleExtend extends SysRole {
    /**
     * 角色包含的权限列表
     */
    private List<SysPrivilege> sysPrivilegeList;

    public List<SysPrivilege> getSysPrivilegeList() {
        return sysPrivilegeList;
    }

    public void setSysPrivilegeList(List<SysPrivilege> sysPrivilegeList) {
        this.sysPrivilegeList = sysPrivilegeList;
    }
}

然后在SysRoleMapper.xml中新建如下映射:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

<resultMap id="rolePrivilegeListMap" extends="roleMap"
           type="com.zwwhnly.mybatisaction.model.SysRoleExtend">
    <collection property="sysPrivilegeList" columnPrefix="privilege_"
                resultMap="com.zwwhnly.mybatisaction.mapper.SysPrivilegeMapper.sysPrivilegeMap"/>
</resultMap>

这里的roleMap我们在之前的博客中已经定义过,代码如下:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

<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>

com.zwwhnly.mybatisaction.mapper.SysPrivilegeMapper.sysPrivilegeMap就是我们刚刚在SysPrivilegeMapper.xml中新建的映射sysPrivilegeMap。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

然后,需要将上篇博客中的userRoleListMap修改为:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

<resultMap id="userRoleListMap" type="com.zwwhnly.mybatisaction.model.SysUserExtend" extends="sysUserMap">
    <collection property="sysRoleList" columnPrefix="role_"
                resultMap="com.zwwhnly.mybatisaction.mapper.SysRoleMapper.rolePrivilegeListMap">
    </collection>
</resultMap>

并且要修改上篇博客中id为selectAllUserAndRoles的select标签代码,因为要关联角色权限关系表和权限表:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

<select id="selectAllUserAndRoles" resultMap="userRoleListMap">
    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_time,
            p.id role_privilege_id,
            p.privilege_name role_privilege_privilege_name,
            p.privilege_url role_privilege_privilege_url
    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
    INNER JOIN sys_role_privilege rp ON rp.role_id = r.id
    INNER JOIN sys_privilege p ON p.id = rp.privilege_id
</select>

注意事项:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

这里sys_privilege表的列名的别名前缀为role_privilege_,这是因为userRoleListMap中collection的columnPrefix属性为role_,并且指定的com.zwwhnly.mybatisaction.mapper.SysRoleMapper.rolePrivilegeListMap中collection的columnPrefix属性为privilege_,所以这里的前缀需要叠加,就变成了role_privilege_文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

3. 单元测试

修改上篇博客中建的测试方法testSelectAllUserAndRoles()代码为:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

@Test
public void testSelectAllUserAndRoles() {
    SqlSession sqlSession = getSqlSession();

    try {
        SysUserMapper sysUserMapper = sqlSession.getMapper(SysUserMapper.class);

        List<SysUserExtend> sysUserList = sysUserMapper.selectAllUserAndRoles();
        System.out.println("用户数:" + sysUserList.size());
        for (SysUserExtend sysUser : sysUserList) {
            System.out.println("用户名:" + sysUser.getUserName());
            for (SysRoleExtend sysRoleExtend : sysUser.getSysRoleList()) {
                System.out.println("角色名:" + sysRoleExtend.getRoleName());
                for (SysPrivilege sysPrivilege : sysRoleExtend.getSysPrivilegeList()) {
                    System.out.println("权限名:" + sysPrivilege.getPrivilegeName());
                }
            }
        }
    } finally {
        sqlSession.close();
    }
}

运行测试代码,测试通过,输出日志如下:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - ==> Preparing: 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_time, p.id role_privilege_id, p.privilege_name role_privilege_privilege_name, p.privilege_url role_privilege_privilege_url 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 INNER JOIN sys_role_privilege rp ON rp.role_id = r.id INNER JOIN sys_privilege p ON p.id = rp.privilege_id文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - ==> Parameters:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Columns: id, user_name, user_password, user_email, create_time, role_id, role_role_name, role_enabled, role_create_by, role_create_time, role_privilege_id, role_privilege_privilege_name, role_privilege_privilege_url文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Row: 1, admin, 123456, admin@mybatis.tk, 2019-06-27 18:21:07.0, 1, 管理员, 1, 1, 2019-06-27 18:21:12.0, 1, 用户管理, /users文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Row: 1, admin, 123456, admin@mybatis.tk, 2019-06-27 18:21:07.0, 1, 管理员, 1, 1, 2019-06-27 18:21:12.0, 2, 角色管理, /roles文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Row: 1, admin, 123456, admin@mybatis.tk, 2019-06-27 18:21:07.0, 1, 管理员, 1, 1, 2019-06-27 18:21:12.0, 3, 系统日志, /logs文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Row: 1, admin, 123456, admin@mybatis.tk, 2019-06-27 18:21:07.0, 2, 普通用户, 1, 1, 2019-06-27 18:21:12.0, 4, 人员维护, /persons文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Row: 1, admin, 123456, admin@mybatis.tk, 2019-06-27 18:21:07.0, 2, 普通用户, 1, 1, 2019-06-27 18:21:12.0, 5, 单位维护, /companies文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

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, 4, 人员维护, /persons文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

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, 5, 单位维护, /companies文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - <== Total: 7文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

用户数:2文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

用户名:admin文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

角色名:管理员文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

权限名:用户管理文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

权限名:角色管理文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

权限名:系统日志文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

角色名:普通用户文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

权限名:人员维护文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

权限名:单位维护文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

用户名:test文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

角色名:普通用户文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

权限名:人员维护文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

权限名:单位维护文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

从日志可以看出,不仅查询出了用户拥有的角色信息,也查询出了角色包含的权限信息。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

4. 延迟加载

有的同学可能会说,返回的角色信息和权限信息我不一定用啊,每次关联这么多表查询一次数据库,好影响性能啊,能不能在我使用到角色信息即获取sysRoleList属性时再去数据库查询呢?答案当然是能,那么如何实现呢?文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

实现延迟加载需要使用collection标签的fetchType属性,该属性有lazy和eager两个值,分别代表延迟加载和积极加载。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

由于需要根据角色Id获取该角色对应的所有权限信息,所以我们要先在SysPrivilegeMapper.xml中定义如下查询:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

<select id="selectPrivilegeByRoleId" resultMap="sysPrivilegeMap">
    SELECT p.*
    FROM sys_privilege p
    INNER JOIN sys_role_privilege rp ON rp.privilege_id = p.id
    WHERE rp.role_id = #{roleId}
</select>

然后在SysRoleMapper.xml中添加如下查询:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

<resultMap id="rolePrivilegeListMapSelect" extends="roleMap"
           type="com.zwwhnly.mybatisaction.model.SysRoleExtend">
    <collection property="sysPrivilegeList" fetchType="lazy"
                column="{roleId=id}"
                select="com.zwwhnly.mybatisaction.mapper.SysPrivilegeMapper.selectPrivilegeByRoleId"/>
</resultMap>
<select id="selectRoleByUserId" resultMap="rolePrivilegeListMapSelect">
    SELECT
          r.id,
          r.role_name,
          r.enabled,
          r.create_by,
          r.create_time
    FROM sys_role r
    INNER JOIN sys_user_role ur ON ur.role_id = r.id
    WHERE ur.user_id = #{userId}
</select>

上面的column="{roleId=id}"中,roleId指的是select指定的方法selectPrivilegeByRoleId的参数,id指的是查询selectRoleByUserId中查询出的角色id。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

然后在SysUserMapper.xml中添加如下查询:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

<resultMap id="userRoleListMapSelect" extends="sysUserMap"
           type="com.zwwhnly.mybatisaction.model.SysUserExtend">
    <collection property="sysRoleList" fetchType="lazy"
                select="com.zwwhnly.mybatisaction.mapper.SysRoleMapper.selectRoleByUserId"
                column="{userId=id}"/>
</resultMap>
<select id="selectAllUserAndRolesSelect" resultMap="userRoleListMapSelect">
    SELECT
          u.id,
          u.user_name,
          u.user_password,
          u.user_email,
          u.create_time
    FROM sys_user u
    WHERE u.id = #{id}
</select>

上面的column="{userId=id}"中,userId指的是select指定的方法selectRoleByUserId的参数,id指的是查询selectAllUserAndRolesSelect中查询出的用户id。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

然后,在SysUserMapper接口中,添加如下方法:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

/**
 * 通过嵌套查询获取指定用户的信息以及用户的角色和权限信息
 *
 * @param id
 * @return
 */
SysUserExtend selectAllUserAndRolesSelect(Long id);

最后,在SysUserMapperTest类中添加如下测试方法:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

@Test
public void testSelectAllUserAndRolesSelect() {
    SqlSession sqlSession = getSqlSession();

    try {
        SysUserMapper sysUserMapper = sqlSession.getMapper(SysUserMapper.class);

        SysUserExtend sysUserExtend = sysUserMapper.selectAllUserAndRolesSelect(1L);
        System.out.println("用户名:" + sysUserExtend.getUserName());
        for (SysRoleExtend sysRoleExtend : sysUserExtend.getSysRoleList()) {
            System.out.println("角色名:" + sysRoleExtend.getRoleName());
            for (SysPrivilege sysPrivilege : sysRoleExtend.getSysPrivilegeList()) {
                System.out.println("权限名:" + sysPrivilege.getPrivilegeName());
            }
        }
    } finally {
        sqlSession.close();
    }
}

运行测试方法,输出日志如下:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - ==> Preparing: SELECT u.id, u.user_name, u.user_password, u.user_email, u.create_time FROM sys_user u WHERE u.id = ?文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - ==> Parameters: 1(Long)文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Columns: id, user_name, user_password, user_email, create_time文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Row: 1, admin, 123456, admin@mybatis.tk, 2019-06-27 18:21:07.0文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - <== Total: 1文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

用户名:admin文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - ==> Preparing: SELECT r.id, r.role_name, r.enabled, r.create_by, r.create_time FROM sys_role r INNER JOIN sys_user_role ur ON ur.role_id = r.id WHERE ur.user_id = ?文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - ==> Parameters: 1(Long)文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Columns: id, role_name, enabled, create_by, create_time文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Row: 1, 管理员, 1, 1, 2019-06-27 18:21:12.0文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Row: 2, 普通用户, 1, 1, 2019-06-27 18:21:12.0文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - <== Total: 2文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

角色名:管理员文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - ==> Preparing: SELECT p.* FROM sys_privilege p INNER JOIN sys_role_privilege rp ON rp.privilege_id = p.id WHERE rp.role_id = ?文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - ==> Parameters: 1(Long)文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Columns: id, privilege_name, privilege_url文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Row: 1, 用户管理, /users文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Row: 2, 角色管理, /roles文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Row: 3, 系统日志, /logs文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - <== Total: 3文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

权限名:用户管理文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

权限名:角色管理文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

权限名:系统日志文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

角色名:普通用户文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - ==> Preparing: SELECT p.* FROM sys_privilege p INNER JOIN sys_role_privilege rp ON rp.privilege_id = p.id WHERE rp.role_id = ?文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - ==> Parameters: 2(Long)文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Columns: id, privilege_name, privilege_url文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Row: 4, 人员维护, /persons文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

TRACE [main] - <== Row: 5, 单位维护, /companies文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

DEBUG [main] - <== Total: 2文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

权限名:人员维护文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

权限名:单位维护文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

仔细分析上面的日志,会发现只有在使用到了角色信息和权限信息时,才执行了对应的数据库查询。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

需要注意的是,延迟加载依赖于MyBatis全局配置中的aggressiveLazyLoading,在之前的博客讲解association标签时,我们已经将其配置为了false,所以这里的执行结果符合我们的预期:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

<settings>
    <!--其他配置-->
    <setting name="aggressiveLazyLoading" value="false"/>
</settings>

关于该参数的详细讲解,请查看使用association标签实现嵌套查询文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

5. 总结

使用collection标签实现嵌套查询,用到的属性总结如下:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

1)select:另一个映射查询的id,MyBatis会额外执行这个查询获取嵌套对象的结果。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

2)column:将主查询中列的结果作为嵌套查询的参数,配置方式如column="{prop1=col1,prop2=col2}",prop1和prop2将作为嵌套查询的参数。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

3)fetchType:数据加载方式,可选值为lazy和eager,分别为延迟加载和积极加载。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

4)如果要使用延迟加载,除了将fetchType设置为lazy,还需要注意全局配置aggressiveLazyLoading的值应该为false。这个参数在3.4.5版本之前默认值为ture,从3.4.5版本开始默认值改为false。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

5)MyBatis提供的lazyLoadTriggerMethods参数,支持在触发某方法时直接触发延迟加载属性的查询,如equals()方法。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/14061.html

  • 本站内容整理自互联网,仅提供信息存储空间服务,以方便学习之用。如对文章、图片、字体等版权有疑问,请在下方留言,管理员看到后,将第一时间进行处理。
  • 转载请务必保留本文链接:https://www.cainiaoxueyuan.com/bc/14061.html

Comment

匿名网友 填写信息

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定