MyBatis-Plus实现跨表查询并返回分页

 更新时间:2026年04月27日 08:35:59   作者:小裕哥略帅  
本文档详细介绍基于MyBatis-Plus框架实现的跨表查询分页功能,以供应商物料查询为例,展示如何通过多表关联查询并返回标准分页对象的技术实现方案

1. 概述

本文档详细介绍基于MyBatis-Plus框架实现的跨表查询分页功能。以供应商物料查询为例,展示如何通过多表关联查询并返回标准分页对象的技术实现方案。

2. 技术栈

ORM框架: MyBatis-Plus

数据库: MySQL(使用LIMIT进行分页)

分页组件: MyBatis-Plus Page对象

架构模式: Repository + Mapper + XML三层架构

3. 核心实现原理

3.1 分页策略

采用手动分页方式,分为两步执行:

查询总数: 执行COUNT查询获取符合条件的总记录数

查询数据: 根据偏移量(offset)和页大小(size)查询具体数据

3.2 为什么选择手动分页

跨表查询涉及LEFT JOIN,MyBatis-Plus自动分页可能存在性能问题

可以精确控制SQL语句,优化查询性能

支持复杂的动态条件拼接

4. 代码实现详解

4.1 参数对象设计

SupplierIdParam.java - 查询参数封装

@Getter
@Setter
@ToString
public class SupplierIdParam extends PageQueryParam {
    private Long tenantId;                        // 租户ID
    private Long supplierId;                      // 供应商ID
    private String materialCategoryExternalCode;  // 物料分类外部编码
    private String materialCategoryName;          // 物料分类名称
    private String materialName;                  // 物料名称
    private String materialExternalCode;          // 物料编码
    private String materialStatus;                // 物料状态
    private String pricingType;                   // 定价类型
    // ... 其他字段
}

关键点:

继承PageQueryParam,包含currentPage和pageSize分页参数

支持多维度动态查询条件

使用包装类型,便于判断参数是否为空

4.2 领域对象设计

SupplierMaterialDO.java - 供应商物料实体

@Getter
@Setter
@TableName(value = "t_supplier_material", autoResultMap = true)
public class SupplierMaterialDO implements BaseEntity {
    private Long id;
    private Long tenantId;
    private Long supplierDataId;
    private Long materialCategoryId;
    private String materialCategoryExternalCode;
    private String materialCategoryName;
    private Long materialId;
    private String externalCode;
    private String name;
    // ... 其他字段
}

关键点:

  • 使用@TableName指定表名
  • autoResultMap = true支持复杂结果映射
  • 实现BaseEntity接口,包含通用字段

4.3 Repository层实现

SupplierMaterialRepository.java

@Repository
public class SupplierMaterialRepository extends HussarServiceImpl<SupplierMaterialMapper, SupplierMaterialDO> {
    @Resource
    private SupplierMaterialMapper supplierMaterialMapper;
    public Page<SupplierMaterialDO> queryMaterialsBySupplierId(SupplierIdParam param) {
        Long tenantId = param.getTenantId();
        // 第一步:查询总数
        Long total = supplierMaterialMapper.queryMaterialsBySupplierIdCount(param, tenantId);
        // 第二步:创建分页对象
        Page<SupplierMaterialDO> page = new Page<>(param.getCurrentPage(), param.getPageSize(), total);
        // 第三步:查询分页数据
        List<SupplierMaterialDO> supplierMaterialDOList = 
            supplierMaterialMapper.queryMaterialsBySupplierId(param, tenantId, page.offset(), page.getSize());
        // 第四步:设置结果集
        if (CollectionUtils.isNotEmpty(supplierMaterialDOList)) {
            page.setRecords(supplierMaterialDOList);
        }
        return page;
    }
}

实现要点:

  • 分离计数查询: 单独执行COUNT查询,避免查询不必要的数据列
  • 计算偏移量: 使用page.offset()方法自动计算OFFSET值
  • 空值保护: 检查查询结果是否为空,避免设置null集合
  • 租户隔离: 所有查询都携带tenantId,实现多租户数据隔离

4.4 Mapper接口定义

SupplierMaterialMapper.java

@Mapper
public interface SupplierMaterialMapper extends BaseMapper<SupplierMaterialDO>, HussarMapper<SupplierMaterialDO> {
    
    /**
     * 查询总数
     */
    Long queryMaterialsBySupplierIdCount(@Param("param") SupplierIdParam param, 
                                         @Param("tenantId") Long tenantId);
    
    /**
     * 查询分页数据
     */
    List<SupplierMaterialDO> queryMaterialsBySupplierId(@Param("param") SupplierIdParam param, 
                                                        @Param("tenantId") Long tenantId, 
                                                        @Param("offset") Long offset, 
                                                        @Param("size") Long size);
}

设计说明:

  • 使用@Param注解明确参数名称,便于XML中引用
  • 将offset和size作为独立参数传递,提高SQL可读性
  • 返回类型分别为Long和List<SupplierMaterialDO>

4.5 MyBatis XML实现

SupplierMaterialMapper.xml

总数查询SQL

<select id="queryMaterialsBySupplierIdCount" resultType="java.lang.Long">
    SELECT count(*) AS total
    FROM t_supplier_material tsm
    LEFT JOIN t_material tm ON tsm.material_id = tm.id
    WHERE tsm.delete_flag = '0'
    AND tsm.tenant_id = #{tenantId}
    <if test="param.supplierId != null and param.supplierId != ''">
        AND supplier_id = #{param.supplierId}
    </if>
    <if test="param.materialCategoryExternalCode != null and param.materialCategoryExternalCode != ''">
        AND tsm.material_category_external_code LIKE CONCAT('%', #{param.materialCategoryExternalCode}, '%')
    </if>
    <if test="param.materialCategoryName != null and param.materialCategoryName != ''">
        AND tsm.material_category_name LIKE CONCAT('%', #{param.materialCategoryName}, '%')
    </if>
    <if test="param.materialName != null and param.materialName != ''">
        AND tsm.name LIKE CONCAT('%', #{param.materialName}, '%')
    </if>
    <if test="param.materialExternalCode != null and param.materialExternalCode != ''">
        AND tsm.external_code LIKE CONCAT('%', #{param.materialExternalCode}, '%')
    </if>
    <if test="param.materialStatus != null and param.materialStatus != ''">
        AND tm.status = #{param.materialStatus}
    </if>
    <if test="param.pricingType != null and param.pricingType != ''">
        AND tm.pricing_type = #{param.pricingType}
    </if>
</select>

分页数据查询SQL

<select id="queryMaterialsBySupplierId"
        resultType="com.mdgyl.hussar.basic.supplier.masterdata.domain.SupplierMaterialDO">
    SELECT tsm.*
    FROM t_supplier_material tsm
    LEFT JOIN t_material tm ON tsm.material_id = tm.id
    WHERE tsm.delete_flag = '0'
    AND tsm.tenant_id = #{tenantId}
    <!-- 动态条件与COUNT查询保持一致 -->
    <if test="param.supplierId != null and param.supplierId != ''">
        AND supplier_id = #{param.supplierId}
    </if>
    <if test="param.materialCategoryExternalCode != null and param.materialCategoryExternalCode != ''">
        AND tsm.material_category_external_code LIKE CONCAT('%', #{param.materialCategoryExternalCode}, '%')
    </if>
    <!-- ... 其他动态条件 ... -->
    ORDER BY tsm.id DESC
    LIMIT #{offset}, #{size}
</select>

SQL编写要点:

  • 表别名规范: 使用简短且有意义的别名(如tsm、tm)
  • 动态条件: 使用<if>标签实现条件可选,确保COUNT和DATA查询条件一致
  • 模糊查询: 使用CONCAT('%', value, '%')实现LIKE模糊匹配
  • 软删除过滤: 始终添加delete_flag = '0'条件
  • 排序规则: 使用ORDER BY id DESC保证数据稳定性
  • LIMIT分页: 使用LIMIT #{offset}, #{size}实现物理分页

到此这篇关于MyBatis-Plus实现跨表查询并返回分页的文章就介绍到这了,更多相关MyBatis-Plus跨表查询分页内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java使用WatchService监控文件内容变化的示例

    Java使用WatchService监控文件内容变化的示例

    本篇文章主要介绍了Java使用WatchService监控文件变化的示例,非常具有实用价值,需要的朋友可以参考下
    2017-10-10
  • Mybatis Plus ActiveRecord模式的具体使用

    Mybatis Plus ActiveRecord模式的具体使用

    ActiveRecord 是一种设计模式,它是一种在软件开发中用于管理关系数据库的模式,本文主要介绍了Mybatis Plus ActiveRecord模式的具体使用,具有一定的参考价值,感兴趣的可以了解一下
    2024-07-07
  • Java对象初始化顺序的使用

    Java对象初始化顺序的使用

    本篇文章介绍了,Java对象初始化顺序的使用。需要的朋友参考下
    2013-04-04
  • knife4j3.0.3整合gateway和注册中心的详细过程

    knife4j3.0.3整合gateway和注册中心的详细过程

    这篇文章主要介绍了knife4j3.0.3整合gateway和注册中心的详细过程,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-03-03
  • Java的MyBatis+Spring框架中使用数据访问对象DAO模式的方法

    Java的MyBatis+Spring框架中使用数据访问对象DAO模式的方法

    Data Access Object数据访问对象模式在Java操作数据库部分的程序设计中经常被使用到,这里我们就来看一下Java的MyBatis+Spring框架中使用数据访问对象DAO模式的方法:
    2016-06-06
  • SpringBoot自动配置原理分析

    SpringBoot自动配置原理分析

    这篇文章主要介绍了SpringBoot自动配置原理分析,SpringBoot是我们经常使用的框架,那么你能不能针对SpringBoot实现自动配置做一个详细的介绍。如果可以的话,能不能画一下实现自动配置的流程图。牵扯到哪些关键类,以及哪些关键点
    2022-08-08
  • 解决Swagger修改请求对象字段文档不更新问题

    解决Swagger修改请求对象字段文档不更新问题

    这篇文章主要为大家介绍了解决Swagger修改请求对象字段文档不更新的问题,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-05-05
  • Springmvc模式上传和下载与enctype对比

    Springmvc模式上传和下载与enctype对比

    这篇文章主要介绍了Springmvc模式上传和下载与enctype对比,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-12-12
  • Java Chassis3负载均衡选择器技术解密

    Java Chassis3负载均衡选择器技术解密

    这篇文章主要为大家介绍了Java Chassis3负载均衡选择器技术解密,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2024-01-01
  • Java自带的Http Server实现设置返回值的类型(content-type)

    Java自带的Http Server实现设置返回值的类型(content-type)

    这篇文章主要介绍了Java自带的Http Server实现设置返回值的类型(content-type),具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-11-11

最新评论