mybatis-plus自带page 实现分页

  • 重点代码

@Service
public class CheckpointService extends ServiceImpl<CheckpointMapper, ControlCheckpoint> {
    Page<ControlCheckpoint> page = this.page(new Page<>(reqVO.getPageNo(), reqVO.getPageSize()), wrapper);
}
  • pom

        <dependency>
            <groupId>com.baomidou</groupId>
            <artifactId>mybatis-plus-boot-starter</artifactId>
            <version>3.5.5</version>
        </dependency>

        <dependency>
            <groupId>com.github.xiaoymin</groupId>
            <artifactId>knife4j-openapi3-spring-boot-starter</artifactId>
            <version>4.1.0</version>
        </dependency>

        <dependency>
            <groupId>org.springdoc</groupId>
            <artifactId>springdoc-openapi-ui</artifactId>
            <version>1.6.15</version>
        </dependency>


        <dependency>
            <groupId>com.github.yulichang</groupId>
            <artifactId>mybatis-plus-join-boot-starter</artifactId>
            <version>1.4.11</version>
        </dependency>
  • controller

/**
 * @Description 卡口管理
 * @Author wpz
 * @Date 2024/06/21 14:04
 */
@Slf4j
@Validated
@RestController
@RequestMapping("/control/checkpoint")
@Tag(name = "卡口管理")
public class ControlCheckpointController {


    @Resource
    private CheckpointService checkpointService;

    @Resource
    private SyncService syncService;


    @GetMapping("/page")
    @Operation(summary = "分页获取卡口管理")
    public CommonResult<PageResult<ControlCheckpoint>> page(@ParameterObject @Valid  ControlCheckpointPageReqVO reqVO) {
        return checkpointService.pageCheckpoint(reqVO);
    }


}
  • service

package com.zygh.investigation.control.service.control;

import cn.hutool.core.bean.BeanUtil;
import cn.hutool.core.collection.CollUtil;
import cn.hutool.core.util.StrUtil;
import com.alibaba.excel.EasyExcel;
import com.baomidou.mybatisplus.core.toolkit.Wrappers;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.github.yulichang.toolkit.JoinWrappers;
import com.github.yulichang.wrapper.MPJLambdaWrapper;
import com.zygh.investigation.control.common.constant.IcsConstants;
import com.zygh.investigation.control.common.pojo.CommonResult;
import com.zygh.investigation.control.common.pojo.PageResult;
import com.zygh.investigation.control.config.mybatis.handler.CheckpointExcelListener;
import com.zygh.investigation.control.dao.control.CheckpointMapper;
import com.zygh.investigation.control.dao.control.ControlGroupEquipmentMapper;
import com.zygh.investigation.control.dao.control.ControlLawEnforceStationMapper;
import com.zygh.investigation.control.dao.control.EmsEquipmentCheckpointMapper;
import com.zygh.investigation.control.dao.control.SsoProjectInfoMapper;
import com.zygh.investigation.control.dto.vehicle.control.CheckpointExcelDTO;
import com.zygh.investigation.control.entity.control.ControlCheckpoint;
import com.zygh.investigation.control.entity.control.ControlGroupEquipment;
import com.zygh.investigation.control.entity.control.ControlStationEquipment;
import com.zygh.investigation.control.entity.control.EmsEquipmentCheckpoint;
import com.zygh.investigation.control.entity.sso.User;
import com.zygh.investigation.control.service.sso.PermissionService;
import com.zygh.investigation.control.vo.control.ControlCheckpointAllVO;
import com.zygh.investigation.control.vo.control.ControlCheckpointPageReqVO;
import com.zygh.investigation.control.vo.control.ControlCheckpointVO;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import org.springframework.web.multipart.MultipartFile;

import javax.annotation.Resource;
import java.io.IOException;
import java.util.Comparator;
import java.util.List;
import java.util.Objects;
import java.util.stream.Collectors;

/**
 * @Description 卡口管理
 * @Author wpz
 * @Date 2024/06/21 14:06
 */
@Slf4j
@Service
public class CheckpointService extends ServiceImpl<CheckpointMapper, ControlCheckpoint> {

    @Resource
    private PermissionService permissionService;

    @Resource
    private CheckpointMapper checkpointMapper;

    @Resource
    private SsoProjectInfoMapper ssoProjectInfoMapper;

    @Resource
    private EmsEquipmentCheckpointMapper emsEquipmentCheckpointMapper;

    @Resource
    private StationEquipmentService stationEquipmentService;

    @Resource
    private ControlLawEnforceStationMapper controlLawEnforceStationMapper;

    @Resource
    private CheckpointGroupEquipmentService checkpointGroupEquipmentService;

    @Resource
    private ControlGroupEquipmentMapper controlGroupEquipmentMapper;

    @Resource
    private SyncService syncService;



    /**
     * 分页获取卡口管理
     *
     * @param reqVO 分页请求
     * @return 分页结果
     */
    public CommonResult<PageResult<ControlCheckpoint>> pageCheckpoint(ControlCheckpointPageReqVO reqVO) {

        List<Integer> projectIdList = CollUtil.newArrayList();

        if (Objects.nonNull(reqVO.getProjectId())) {
            projectIdList = ssoProjectInfoMapper.getSonIds(reqVO.getProjectId());
        }

        PageResult<ControlCheckpoint> pageResult = new PageResult<>();

        MPJLambdaWrapper<ControlCheckpoint> wrapper = JoinWrappers.lambda(ControlCheckpoint.class)
                .selectAll()
                .selectAs(EmsEquipmentCheckpoint::getAssetParentName, ControlCheckpoint::getAssetParentName)
                .selectAs(EmsEquipmentCheckpoint::getAssetChildName, ControlCheckpoint::getAssetChildName)
                .selectAs(EmsEquipmentCheckpoint::getProjectName, ControlCheckpoint::getProjectName)
                .selectAs(EmsEquipmentCheckpoint::getName, ControlCheckpoint::getName)
                .innerJoin(EmsEquipmentCheckpoint.class, EmsEquipmentCheckpoint::getCode, ControlCheckpoint::getCode)
                .orderByDesc(ControlCheckpoint::getId);

        if (Objects.nonNull(reqVO.getProjectId())) {
            wrapper.in(EmsEquipmentCheckpoint::getProjectId, projectIdList);
        }

        if (Objects.nonNull(reqVO.getAssetParent())) {
            wrapper.eq(EmsEquipmentCheckpoint::getAssetParent, reqVO.getAssetParent());
        }

        if (Objects.nonNull(reqVO.getAssetChild())) {
            wrapper.eq(EmsEquipmentCheckpoint::getAssetChild, reqVO.getAssetChild());
        }

        if (Objects.nonNull(reqVO.getStatus())) {
            wrapper.eq(ControlCheckpoint::getStatus, reqVO.getStatus());
        }

        if (StrUtil.isNotBlank(reqVO.getKeyword())) {
            wrapper.and(e -> e.like(ControlCheckpoint::getCode, reqVO.getKeyword())
                    .or().like(EmsEquipmentCheckpoint::getName, reqVO.getKeyword()));
        }


        Page<ControlCheckpoint> page = this.page(new Page<>(reqVO.getPageNo(), reqVO.getPageSize()), wrapper);

        pageResult.setList(page.getRecords());
        pageResult.setTotal(page.getTotal());

        return CommonResult.success(pageResult);
    }
}

ControlCheckpointPageReqVO--- 请求参数

package com.zygh.investigation.control.vo.control;

import com.zygh.investigation.control.common.pojo.PageParam;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data;
import lombok.EqualsAndHashCode;

/**
 * @Description 卡口管理分页请求VO
 * @Author wpz
 * @Date 2024/06/22 11:51
 */

@EqualsAndHashCode(callSuper = true)
@Data
@Schema(description = "卡口管理分页请求VO")
public class ControlCheckpointPageReqVO extends PageParam {


    /**
     * 名称,编号模糊搜索
     */
    @Schema(description = "名称,编号模糊搜索")
    private String keyword;

    /**
     * 项目ID
     */
    @Schema(description = "项目ID")
    private Integer projectId;

    /**
     * 资产大类Id
     */
    @Schema(description = "资产大类Id")
    private Integer assetParent;

    /**
     * 资产小类Id
     */
    @Schema(description = "资产小类Id")
    private Integer assetChild;


    /**
     * 状态:0-启用;1-禁用
     */
    @Schema(description = "状态:0-启用;1-禁用")
    private Integer status;




}

PageParam

package com.zygh.investigation.control.common.pojo;

import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data;

import javax.validation.constraints.Max;
import javax.validation.constraints.Min;
import javax.validation.constraints.NotNull;
import java.io.Serializable;

@Schema(description="分页参数")
@Data
public class PageParam implements Serializable {

    private static final Integer PAGE_NO = 1;
    private static final Integer PAGE_SIZE = 10;

    @Schema(description = "页码,从 1 开始", requiredMode = Schema.RequiredMode.REQUIRED)
    @NotNull(message = "页码不能为空")
    @Min(value = 1, message = "页码最小值为 1")
    private Integer pageNo = PAGE_NO;

    @Schema(description = "每页条数,最大值为 100", requiredMode = Schema.RequiredMode.REQUIRED)
    @NotNull(message = "每页条数不能为空")
    @Min(value = 1, message = "每页条数最小值为 1")
    @Max(value = 100, message = "每页条数最大值为 100")
    private Integer pageSize = PAGE_SIZE;

}
  • PageResult---响应信息

package com.zygh.investigation.control.common.pojo;

import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data;

import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;

/**
 * @author liubh
 */
@Schema(description = "分页结果")
@Data
public final class PageResult<T> implements Serializable {

    @Schema(description = "数据", requiredMode = Schema.RequiredMode.REQUIRED)
    private List<T> list;

    @Schema(description = "总量", requiredMode = Schema.RequiredMode.REQUIRED)
    private Long total;

    public PageResult() {
    }

    public PageResult(List<T> list, Long total) {
        this.list = list;
        this.total = total;
    }

    public PageResult(Long total) {
        this.list = new ArrayList<>();
        this.total = total;
    }

    public static <T> PageResult<T> empty() {
        return new PageResult<>(0L);
    }

    public static <T> PageResult<T> empty(Long total) {
        return new PageResult<>(total);
    }

}
  • ControlCheckpoint

package com.zygh.investigation.control.entity.control;

import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableField;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableName;
import com.zygh.investigation.control.config.mybatis.dataobject.BaseDO;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.experimental.Accessors;

/**
 * 卡口管理表
 *
 * @author admin
 * @TableName control_checkpoint
 */
@EqualsAndHashCode(callSuper = true)
@TableName(value = "control_checkpoint")
@Data
@Accessors(chain = true)
public class ControlCheckpoint extends BaseDO {
    /**
     * id
     */
    @Schema(description = "id")
    @TableId(type = IdType.AUTO)
    private Long id;

    /**
     * 设备编号
     */
    @Schema(description = "设备编号")
    private String code;

    /**
     * 状态:0-启用;1-禁用
     */
    @Schema(description = "状态:0-启用;1-禁用")
    private Integer status;

    /**
     * 部门id
     */
    @Schema(description = "部门id", hidden = true)
    private String deptId;

    /**
     * 添加人id
     */
    @Schema(description = "添加人id", hidden = true)
    private Long createUser;

    /**
     * 添加人名称
     */
    @Schema(description = "添加人名称")
    private String createUserName;

    /**
     * 类型:0-新增;1-导入
     */
    @Schema(description = "类型:0-新增;1-导入", hidden = true)
    private Integer type;


    @Schema(description = "项目名称")
    @TableField(exist = false)
    private String projectName;

    @Schema(description = "资产大类名称")
    @TableField(exist = false)
    private String assetParentName;

    @Schema(description = "资产小类名称")
    @TableField(exist = false)
    private String assetChildName;

    @Schema(description = "设备名称")
    @TableField(exist = false)
    private String name;

    /**
     * 站id
     */
    @Schema(description = "站id")
    @TableField(exist = false)
    private Long stationId;

    /**
     * 组id
     */
    @Schema(description = "组id")
    @TableField(exist = false)
    private Long groupId;
}
  • BaseDO

package com.zygh.investigation.control.config.mybatis.dataobject;

import com.baomidou.mybatisplus.annotation.FieldFill;
import com.baomidou.mybatisplus.annotation.TableField;
import com.baomidou.mybatisplus.extension.activerecord.Model;
import com.fasterxml.jackson.annotation.JsonFormat;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data;

import java.time.LocalDateTime;

/**
 * 基础实体对象
 *
 * @author zygh
 */
@Data
public abstract class BaseDO extends Model<BaseDO> {

    /**
     * 创建时间
     */
    @TableField(fill = FieldFill.INSERT)
    @Schema(description = "创建时间")
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
    private LocalDateTime gmtCreate;
    /**
     * 最后更新时间
     */
    @TableField(fill = FieldFill.INSERT_UPDATE)
    @Schema(description = "更新时间", hidden = true)
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
    private LocalDateTime gmtUpdate;

}

<think>嗯,我现在需要了解MyBatis-Plus是如何实现分页的。我之前用过MyBatis,知道它本身没有分页功能,需要自己写Limit或者用PageHelper这样的插件。MyBatis-Plus作为增强工具,应该在这方面有封装吧。 首先,我应该回忆一下MyBatis-Plus的基本用法。记得在配置类里可能会添加分页插件,比如PaginationInterceptor或者新的MybatisPlusInterceptor,对吗?可能不同版本有不同的配置方式。比如,旧版本用PaginationInterceptor,而新版本可能需要用MybatisPlusInterceptor,然后添加分页的插件进去。 然后,分页查询的时候,可能需要创建一个Page对象,传入当前页和每页大小。接着,调用Mapper的selectPage方法,或者Service层的page方法。这里可能需要用到IPage接口,Page是它的实现类。返回的结果会包含分页的数据和总条数等信息。 另外,分页的原理是什么?MyBatis-Plus应该是在底层自动生成了分页的SQL语句,比如MySQL的LIMIT。那它是如何拦截原来的SQL,然后进行改造的呢?可能通过插件机制,拦截执行的SQL,计算分页参数,然后拼接上LIMIT和OFFSET。同时,还会执行一个COUNT查询来获取总记录数。 对了,还要注意是否需要配置分页插件,如果不配置的话,分页可能不生效。比如,在Spring Boot项目中,需要在配置类里添加@Bean方法,注册分页插件。例如: @Bean public MybatisPlusInterceptor mybatisPlusInterceptor() { MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor(); interceptor.addInnerInterceptor(new PaginationInnerInterceptor()); return interceptor; } 这样配置之后,才能使用分页功能。 另外,分页查询是否需要自定义XML或者可以使用Wrapper?好像大部分时候可以直接用selectPage方法,传入Page对象和Wrapper条件,而不需要写SQL。但如果复杂的查询可能需要自己写XML,但分页参数会自动处理。 可能还有一些细节,比如数据库类型不同的处理,比如Oracle的分页和MySQL的分页语法不同,MyBatis-Plus是如何适配的?应该是通过PaginationInnerInterceptor根据配置的数据库类型来生成不同的分页SQL。 那用户使用的时候,步骤大概是: 1. 配置分页插件。 2. 创建Page对象,设置当前页和大小。 3. 调用Mapper或Service的分页方法,传入Page和查询条件。 4. 处理返回的Page对象,获取数据和分页信息。 有没有可能自定义分页的SQL?比如在XML里写自己的查询语句,但参数使用Page,这时MyBatis-Plus会自动处理分页和总数量的查询。需要确保在XML中的查询id对应的方法参数包含Page类型,或者在接口方法返回IPage类型。 另外,分页查询的性能问题,当数据量大的时候,COUNT操作可能会慢,这时候有没有优化方法?比如,某些情况下可以跳过COUNT查询,或者自定义COUNT语句。 可能还要注意,当使用Wrapper时,分页的条件会被自动附加到查询中,同时COUNT语句也会应用同样的条件,以确保总数正确。 总结一下,MyBatis-Plus分页实现主要依赖于分页插件拦截SQL,自动拼接分页语句,并执行COUNT查询,然后通过IPage对象返回结果。用户只需简单配置和调用方法即可,无需手动处理分页逻辑。</think>MyBatis-Plus分页功能通过 **插件拦截+自动SQL优化** 实现,以下是具体实现步骤和原理: --- ### 一、配置分页插件 需在Spring Boot配置类中注册分页拦截器(不同版本配置方式略有差异): ```java @Configuration public class MybatisPlusConfig { // MyBatis-Plus 3.4+ 版本配置 @Bean public MybatisPlusInterceptor mybatisPlusInterceptor() { MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor(); // 添加分页拦截器,并指定数据库类型(如MySQL) interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL)); return interceptor; } } ``` --- ### 二、核心实现原理 1. **拦截SQL** 分页拦截器(`PaginationInnerInterceptor`)会拦截待执行的SQL语句。 2. **自动生成分页SQL** 根据当前数据库类型(如MySQL的`LIMIT`、Oracle的`ROWNUM`),在原SQL基础上动态拼接分页条件。 3. **执行COUNT查询** 自动生成并执行`SELECT COUNT(1)`语句,获取总记录数。 --- ### 三、使用分页功能 #### 方式1:Mapper层直接调用 ```java // 创建分页参数:当前页=1,每页10条 Page<User> page = new Page<>(1, 10); // 执行分页查询(可附加查询条件) Page<User> result = userMapper.selectPage(page, null); // 获取数据 List<User> users = result.getRecords(); // 当前页数据 long total = result.getTotal(); // 总记录数 ``` #### 方式2:Service层调用 ```java // 使用Service的分页方法 IPage<User> page = userService.page(new Page<>(1, 10), Wrappers.<User>query().eq("age", 25)); ``` --- ### 四、自定义SQL分页 若需在XML中编写自定义SQL,只需满足以下条件: 1. 方法第一个参数为`Page<?>`类型 2. 返回值类型为`IPage<?>` ```xml <!-- XML映射文件 --> <select id="selectUserByPage" resultType="User"> SELECT * FROM user WHERE age > #{age} </select> ``` ```java // Mapper接口 IPage<User> selectUserByPage(Page<User> page, @Param("age") Integer age); // 调用方式 Page<User> pageParam = new Page<>(1, 10); IPage<User> result = userMapper.selectUserByPage(pageParam, 18); ``` --- ### 五、高级特性 1. **关闭COUNT查询** ```java Page<User> page = new Page<>(1, 10).setSearchCount(false); // 不执行SELECT COUNT ``` 2. **自定义COUNT语句** 在XML中定义`id="selectCustomPage_count"`的查询,拦截器会自动匹配。 3. **多租户/性能优化** 分页拦截器与其他拦截器(如租户拦截器)协同工作,确保SQL正确性。 --- ### 六、注意事项 - **版本兼容性**:3.4+版本使用`MybatisPlusInterceptor`,旧版本用`PaginationInterceptor`。 - **性能问题**:大数据量表COUNT查询较慢,建议通过业务层缓存优化。 - **复杂SQL**:联表查询时需手动编写XML,但分页逻辑仍由插件自动处理。 通过上述机制,MyBatis-Plus 实现了简洁高效的分页功能,开发者只需关注业务逻辑,无需手动处理分页细节。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值