Mybatis Plus一对多联表查询及分页解决方案

Posted 沙糖橘

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Mybatis Plus一对多联表查询及分页解决方案相关的知识,希望对你有一定的参考价值。

文章目录

需求

查询用户信息列表,其中包含用户对应角色信息,页面检索条件有根据角色名称查询用户列表;

需求分析

一个用户对应多个角色,用户信息和角色信息分表根据用户id关联存储,用户和角色一对多进行表连接查询,

创建对应表:

CREATE TABLE `sys_user` (
  `id` bigint NOT NULL AUTO_INCREMENT COMMENT '用户ID',
  `name` varchar(50) DEFAULT NULL COMMENT '姓名',
  `age` int DEFAULT NULL COMMENT '年龄',
  PRIMARY KEY (`id`) USING BTREE
) ENGINE=InnoDB  DEFAULT CHARSET=utf8mb4  COMMENT='用户信息表';

CREATE TABLE `sys_role` (
  `id` bigint NOT NULL AUTO_INCREMENT COMMENT '角色ID',
  `role_name` varchar(30) NOT NULL COMMENT '角色名称',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB  DEFAULT CHARSET=utf8mb4  COMMENT='角色信息表';


CREATE TABLE `sys_user_role` (
  `id` bigint NOT NULL AUTO_INCREMENT COMMENT 'ID',
  `user_id` bigint NOT NULL COMMENT '用户ID',
  `role_id` bigint NOT NULL COMMENT '角色ID',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4  COMMENT='用户和角色关联表';


INSERT INTO tsq.sys_user (name,age) VALUES
	 ('张三',18),
	 ('王二',19);

INSERT INTO tsq.sys_role (role_name) VALUES
	 ('角色1'),
	 ('角色2'),
	 ('角色3'),
	 ('角色4');

INSERT INTO tsq.sys_user_role (user_id,role_id) VALUES
	 (1,1),
	 (1,2),
	 (1,3),
	 (2,4);


对应实体类:

@Data
@ApiModel("用户信息表")
@TableName("sys_user")
public class User implements Serializable 

    private static final long serialVersionUID = 1L;
    
    @ApiModelProperty("用户id")
    private Long id;
    @ApiModelProperty("姓名")
    private String name;
    @ApiModelProperty("年龄")
    private Integer age;



@Data
@ApiModel("角色信息表")
@TableName("sys_role")
public class Role implements Serializable 

    private static final long serialVersionUID = 1L;
    
    @ApiModelProperty("角色id")
    private Long id;
    @ApiModelProperty("角色名称")
    private String roleName;




@Data
@ApiModel("用户信息表")
public class UserVo implements Serializable 

    private static final long serialVersionUID = 1L;

    @ApiModelProperty("用户id")
    private Long id;
    @ApiModelProperty("姓名")
    private String name;
    @ApiModelProperty("年龄")
    private Integer age;
    
    private List<Role> roleList;




分页问题说明

在使用一对多连接查询并且分页时,发现返回的分页列表数据数量不对
比如这里查询用户对应角色列表,如果使用直接映射,那么 roleList 的每个 Role 对象都会算一条数据;比如查第一页,一个用户有三个角色每页三条数据,就会出现查出一个 User ,三个 Role 的这些情况,这它也算每页三条(其实就只查到一个用户)

分页问题原因

mybatis-plus一对多分页时,应该使用子查询的映射方式,使用直接映射就会出错
所以直接映射适用于一对一,子查询映射使用于一对多;

一对多场景一

查询用户表的内容,角色表不参与条件查询,用懒加载形式

// controller
  @GetMapping("/pageList")
	public Map<String, Object> pageList(@RequestParam(required = false, defaultValue = "0") int offset,
	                                    @RequestParam(required = false, defaultValue = "10") int pagesize) 
		return userService.pageList(offset, pagesize);
	
  
  
// serviceimpl
  @Override
	public Map<String, Object> pageList(int offset, int pagesize) 
      List<UserVo> pageList = userMapper.pageList(offset, pagesize);
      int totalCount = userMapper.pageListCount();
      Map<String, Object> result = new HashMap<String, Object>();
      result.put("pageList", pageList);
      result.put("totalCount", totalCount);

      return result;
	
  
// mapper.xml

   <resultMap id="getUserInfo" type="com.tsq.democase.onetomany.domain.vo.UserVo" >
        <result column="id" property="id" />
        <result column="name" property="name" />
        <result column="age" property="age" />
        <collection property="roleList" javaType="ArrayList" ofType="com.tsq.democase.onetomany.domain.Role"
                    select="getRolesByUserId" column="userId = id"/>
    </resultMap>

    <select id="getRolesByUserId" resultType="com.tsq.democase.onetomany.domain.Role">
        SELECT *
        FROM sys_user_role ur
                 inner join sys_role r on ur.role_id = r.id
        where ur.user_id = #userId
    </select>
    
    <select id="pageList"  resultMap="getUserInfo">
        SELECT *
        FROM sys_user
        LIMIT #offset, #pageSize
    </select>
    
    <select id="pageListCount" resultType="java.lang.Integer">
        SELECT count(1)
        FROM sys_user
    </select>

查询结果

一对多场景二

查询用户表的内容,角色表要作为查询条件参与查询,例如要根据角色名称查询出用户列表

// controller
	@GetMapping("/pageListByRoleName")
	public Map<String, Object> pageListByRoleName(@RequestParam(required = false, defaultValue = "0") int offset,
	                                    @RequestParam(required = false, defaultValue = "10") int pagesize,
	                                    @RequestParam String roleName) 
		return userService.pageListByRoleName(offset, pagesize, roleName);
	
  
  
// serviceimpl
	@Override
	public Map<String, Object> pageListByRoleName(int offset, int pagesize,String roleName) 
      List<UserVo> pageList = userMapper.pageListByRoleName(offset, pagesize, roleName);
      int totalCount = userMapper.pageListCount();
      Map<String, Object> result = new HashMap<String, Object>();
      result.put("pageList", pageList);
      result.put("totalCount", totalCount);

      return result;
	
  
// mapper.xml
   <resultMap id="getUserInfoByRoleName" type="com.tsq.democase.onetomany.domain.vo.UserVo" >
        <result column="id" property="id" />
        <result column="name" property="name" />
        <result column="age" property="age" />
        <collection property="roleList" javaType="ArrayList" ofType="com.tsq.democase.onetomany.domain.Role"
                    select="getRolesByUserIdAndRoleName" column="userId = id,roleName = roleName"/>
    </resultMap>

    <select id="getRolesByUserIdAndRoleName" resultType="com.tsq.democase.onetomany.domain.Role">
        SELECT *
        FROM sys_user_role ur
        inner join sys_role r on ur.role_id = r.id
        where ur.user_id = #userId
        <if test="roleName != null and roleName != ''" >
            and r.role_name LIKE concat('%', #roleName, '%')
        </if>
    </select>

    <select id="pageListByRoleName"  resultMap="getUserInfoByRoleName">
        SELECT temp.* FROM (
        SELECT distinct u.*,#roleName as roleName
        FROM sys_user u
        left join sys_user_role ur on u.id = ur.user_id
        left join sys_role r on r.id = ur.role_id
        <where>
            <if test="roleName != null and roleName != ''" >
                r.role_name LIKE concat('%', #roleName, '%')
            </if>
        </where>
        ) temp
        LIMIT #offset, #pageSize
    </select>

查询结果

性能优化

原因:

场景一二中使用 select方式会触发多次子查询(SELECT *FROM sys_user_role ur inner join sys_role …),当数据量大时会使查询速度很慢。

场景二中查询时产生的sql日志如下:

--  ==>  
SELECT
    temp.* 
FROM
    ( SELECT
        distinct u.*,
        '角色' as roleName 
    FROM
        sys_user u 
    left join
        sys_user_role ur 
            on u.id = ur.user_id 
    left join
        sys_role r 
            on r.id = ur.role_id 
    WHERE
        r.role_name LIKE concat('%', '角色', '%') ) temp LIMIT 0,
    10 
 --  ====>  
SELECT
    * 
FROM
    sys_user_role ur 
inner join
    sys_role r 
        on ur.role_id = r.id 
where
    ur.user_id = 1 
    and r.role_name LIKE concat('%', '角色', '%') 
 --  ====>  
SELECT
    * 
FROM
    sys_user_role ur 
inner join
    sys_role r 
        on ur.role_id = r.id 
where
    ur.user_id = 2 
    and r.role_name LIKE concat('%', '角色', '%') 
 --  ==>  
SELECT
    count(1) 
FROM
    sys_user 

sql可见如果有100各用户就要执行一百次子查询,效率极低。

优化解决方案

sql中只查询sys_user相关信息并且做roleName 过滤,roleList在java代码中用stream关联role并赋值roleList;

// serviceimpl
  @Override
  public Map<String, Object> pageListByRoleName(int offset, int pagesize,String roleName) 
     // List<UserVo> pageList = userMapper.pageListByRoleName(offset, pagesize, roleName);
     List<UserVo> pageList = userMapper.pageListByRoleName2(offset, pagesize, roleName);
     List<Long> userIds = pageList.stream().map(UserVo::getId).collect(Collectors.toList());
     List<UserRoleVo> userRoleVos =  userMapper.getUserRoleByUserIds(userIds);
     Map<Long, List<UserRoleVo>> userRoleMap = userRoleVos.stream().collect(Collectors.groupingBy(UserRoleVo::getUserId, Collectors.toList()));
     pageList.forEach(u -> 
        List<UserRoleVo> roleVos = userRoleMap.get(u.getId());
        List<RoleVo> roles = BeanUtils.listCopy(roleVos, CopyOptions.create(), RoleVo.class);
        u.setRoleList(roles);
     );
     int totalCount = userMapper.pageListCount();
     Map<String, Object> result = new HashMap<String, Object>();
     result.put("pageList", pageList);
     result.put("totalCount", totalCount);

     return result;
  

// mapper.xml
<select id="pageListByRoleName2"  resultType="com.tsq.democase.onetomany.domain.vo.UserVo">
    SELECT distinct u.*
    FROM sys_user u
    left join sys_user_role ur on u.id = ur.user_id
    left join sys_role r on r.id = ur.role_id
    <where>
        <if test="roleName != null and roleName != ''" >
            r.role_name LIKE concat('%', #roleName, '%')
        </if>
    </where>
    LIMIT #offset, #pageSize
</select>  

查询结果

同场景二。

查询时产生的sql如下:

  --  ==>  
  SELECT
      distinct u.* 
  FROM
      sys_user u 
  left join
      sys_user_role ur 
          on u.id = ur.user_id 
  left join
      sys_role r 
          on r.id = ur.role_id 
  WHERE
      r.role_name LIKE concat('%', '角色', '%') LIMIT 0, 10 
   --  ==>  
  SELECT
      ur.user_id ,
      r.id roleId,
      r.role_name 
  FROM
      sys_user_role ur 
  inner join
      sys_role r 
          on ur.role_id = r.id 
   --  ==>  
  SELECT
      count(1) 
  FROM
      sys_user 
 

由sql日志可见这种方式比纯sql方式效率高一些

MyBatis-Plus 不支持联表?一个依赖轻松搞定,非常牛逼!

点击关注公众号,实用技术文章及时了解

1、什么是Mybatis-plus-join?

Mybatis-plus-join是网上的大佬为解决Mybatis-plus做多表连接查询吃力而开发的一个工具,它能基于Mybatisplus进行功能升级,使其不再受限于单表查询,其中本人总结有以下几点:

  • 对Mybatis-plus进行功能升级 ,提高开发效率;

  • 使用方法与Mybatis-plus一样 ,学习成本低;

  • 增加了 多表连接查询 功能,摆脱xml模式多表连接;

2、下载Mybatis-plus-join

Mybatis-plus-join目前发布在Gitee上,直接将其克隆/下载到本地即可:

下载地址:

  • https://gitee.com/best_handsome/mybatis-plus-join

注意:需要 Mybatis-plus version >= 3.4.0

下载结果如图:

3、使用Maven将其打包成jar包

3.1、使用idea打包

1.按照下图进行打包操作:

2.打包成功后,jar包存在的目录,找到jar包并复制到自己的项目中:

3.将打出来的jar包引入工程模块中,步骤如图(如果跳出版本问题,则点击更新即可):

4.项目中引用Mybatis-plus-join的pom.xml配置

在pom.xml中引入该依赖:

<!--mybatis-plus-join-->
<dependency>
 <groupId>com.github.yulichang</groupId>
 <artifactId>mybatis-plus-join</artifactId>
 <version>1.1.6</version>
</dependency>

引入效果(检查时提示更新就更新部分环境即可)

5、让mybatis-plus-join在DataScopeSqlInjector中生效

1.修改DataScopeSqlInjector中的继承类为:MPJSqlInjector

6、启动类排除MPJSqlInjector.class

排除MPJSql注入器的语句

@SpringBootApplication(exclude = MPJSqlInjector.class)

1.在主启动类中排除MPJ的sql注入器

2.如果自定义了配置类,则在配置类中添加排除

7.检查启用mybatis-plus-join是否成功

  1. 重启启动类

  2. 正常启动则说明配置成功,请转到第8步

  3. 如果出现下列错误,说明第6步没有配置好,要检查

***************************
APPLICATION FAILED TO START
***************************
Description:
file [D:package\\admin\\mapper\\SysOauthClientDetailsMapper.class] required a single bean, but 2 were found:
 - dataScopeSqlInjector: defined by method 'dataScopeSqlInjector' in class path resource [package/common/data/mybatis/MybatisPlusConfiguration.class]
 - com.github.yulichang.injector.MPJSqlInjector: defined in null
 - 
Action:
Consider marking one of the beans as @Primary, updating the consumer to accept multiple beans, or using @Qualifier to identify the bean that should be consumed

8、在业务中使用mybatis-plus-join的准备

使用

  • mapper继承MPJBaseMapper (必选)

  • service继承MPJBaseService (可选)

  • serviceImpl继承MPJBaseServiceImpl (可选)

继承效果如下

  1. mapper

  1. serviceImpl

9.使用mybatis-plus-join进行多表查询(例子)

1.DTO类

package *.admin.api.dto;

import cn.afterturn.easypoi.excel.annotation.Excel;
import com.baomidou.mybatisplus.annotation.TableId;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;
import java.math.BigDecimal;
import java.time.LocalDateTime;

@Data
public class ObOutPoolDto  
 /**
  * 表1:物品表
  */
 private String name;  //物品名称
 private String standard; //物品规格
 private String unit;  //物品单位
 private BigDecimal price;  //物品价格
 private BigDecimal count;//物品库存数量
 /**
  * 表2:订单表
  */
 private String drawNumber; //表单流水号
 public Integer poolId; //库存id
 private Integer needCount;//出库数量
 private String person;//申请人
 private LocalDateTime takeTime;//任务时间
 private String department;//部门
 private String updatedBy;//更新人
 private LocalDateTime updatedTime;//更新时间
 private Integer id;//id
 private String remark;//备注

2.serviceImpl

package *
import com.baomidou.mybatisplus.core.metadata.IPage;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.github.yulichang.base.MPJBaseServiceImpl;
import com.github.yulichang.wrapper.MPJLambdaWrapper;
import org.springframework.stereotype.Service;
import java.util.List;

/**
 * 出库商品
 * @author 
 * @date 2021-04-23 14:25:26
 */
@Service
public class ObOutPoolItemServiceImpl extends MPJBaseServiceImpl<ObOutPoolItemMapper, ObOutPoolItem> implements ObOutPoolItemService 
 // 定义一个枚举类型
 enum  Signal 
  FINISH("完成",1),ORDER("订单",2);
  private String name;
  private int index;
  // 构造方法
  private Signal(String name,int index) 
   this.name = name;
   this.index = index;
  
 

 /**
  * 外部jar包支持的连接查询并分页,弥补mytabisplus的不足
  * 分页查询
  */
 @Override
 public IPage<ObOutPoolDto> mypage(Page page, ObOutPoolItem obOutPoolItem, String[] createdTimeArr)
  IPage<ObOutPoolDto> page1 = baseMapper.selectJoinPage(
    page,   //分页参数
    ObOutPoolDto.class,   //返回的DTO层数据,
    mpjLambdaWrapper(MPJ(),obOutPoolItem,createdTimeArr,Signal.FINISH.name));  //MPJWrapper
  return page1;
 

 /**
  * 按照订单号查询商品
  * @param orderNumber
  * @return
  */
 @Override
 public List<ObOutPoolDto> queryByOrderNumber(String orderNumber)
  List<ObOutPoolDto> list = baseMapper.selectJoinList(
    ObOutPoolDto.class,
    mpjLambdaWrapper(MPJ(),new ObOutPoolItem() ,new String[0] ,Signal.ORDER.name,orderNumber));
  return list;
 

 //基础连接查询MPJ对象
 @Override
 public MPJLambdaWrapper<ObOutPoolDto> MPJ()
  MPJLambdaWrapper<ObOutPoolDto> mpj =  new MPJLambdaWrapper<ObOutPoolDto>()
    .selectAll(ObOutPoolItem.class)    //查询表1所有内容
    .selectAll(ObPool.class)           //查询表2所有内容
    .leftJoin(ObPool.class,ObPool::getId,ObOutPoolItem::getPoolId)  //左连接表1,条件为表1库存id 等于 表2 id
         .leftJoin(ObOutPool.class,ObOutPool::getDrawNumber,ObOutPoolItem::getDrawNumber);  //左连接表2,用订单号匹配
  return mpj;
 
 /**
  * 根据需求模糊生成条件
  *后期生成泛型使用类替代此方法
  */
 private MPJLambdaWrapper<ObOutPoolDto> mpjLambdaWrapper(
   MPJLambdaWrapper<ObOutPoolDto> mpj, ObOutPoolItem obOutPoolItem,String[] createdTimeArr,String... status
 )

  //模糊查询部门
  if(!(obOutPoolItem.getDepartment()==null))
   mpj.likeRight(ObOutPoolItem::getDepartment,
     obOutPoolItem.getDepartment());
  
  // 模糊匹配时间
  if(createdTimeArr != null && createdTimeArr.length > 0 && createdTimeArr[0].length() > 0)
   mpj.ge(ObOutPoolItem::getCreatedTime,createdTimeArr[0])
     .le(ObOutPoolItem::getCreatedTime,createdTimeArr[1]);
  
  //完成与订单号查询
  if(status[0].equals(Signal.FINISH.name))
              mpj.eq(ObOutPool::getTaskStatus, TaskStatusEnum.COMPLETED.getStatus());  //完成条件
  else if (status[0].equals(Signal.ORDER.name))
   mpj.eq(ObOutPool::getDrawNumber,status[1]);
  
  return mpj;
 

3.接口返回结果

10、mybatis-plus-join更多使用方法请参看Gitee

  • https://gitee.com/best_handsome/mybatis-plus-join

来源:blog.csdn.net/qq_44140499/article/details/116517163

推荐

Java面试题宝典

技术内卷群,一起来学习!!

PS:因为公众号平台更改了推送规则,如果不想错过内容,记得读完点一下“在看”,加个“星标”,这样每次新文章推送才会第一时间出现在你的订阅列表里。点“在看”支持我们吧!

以上是关于Mybatis Plus一对多联表查询及分页解决方案的主要内容,如果未能解决你的问题,请参考以下文章

SQL 一对多联表查询最大值

MyBatis03:ResultMap及分页

Mybatis实现oracle批量插入及分页查询

MyBatis-Plus 不支持联表?一个依赖轻松搞定,非常牛逼!

MyBatis-Plus联表查询的短板,终于有一款工具补齐了

MyBatis-Plus联表查询的短板,终于有一款工具补齐了