Mybatis拦截器分页

这里做的是以ByPage结尾的,使用mysql数据库
一、page类:

package com.ming.util;

/**
 * 分页对应的实体类
 */
public class Page {
    /**
     * 总条数
     */
    private int totalNumber;
    /**
     * 当前第几页
     */
    private int currentPage;
    /**
     * 总页数
     */
    private int totalPage;
    /**
     * 每页显示条数
     */
    private int pageNumber = 5;
    /**
     * 数据库中limit的参数,从第几条开始取
     */
    private int dbIndex;
    /**
     * 数据库中limit的参数,一共取多少条
     */
    private int dbNumber;

    /**
     * 根据当前对象中属性值计算并设置相关属性值
     */
    public void count() {
        // 计算总页数
        int totalPageTemp = this.totalNumber / this.pageNumber;
        int plus = (this.totalNumber % this.pageNumber) == 0 ? 0 : 1;
        totalPageTemp = totalPageTemp + plus;
        if(totalPageTemp <= 0) {
            totalPageTemp = 1;
        }
        this.totalPage = totalPageTemp;

        // 设置当前页数
        // 总页数小于当前页数,应将当前页数设置为总页数
        if(this.totalPage < this.currentPage) {
            this.currentPage = this.totalPage;
        }
        // 当前页数小于1设置为1
        if(this.currentPage < 1) {
            this.currentPage = 1;
        }

        // 设置limit的参数
        this.dbIndex = (this.currentPage - 1) * this.pageNumber;
        this.dbNumber = this.pageNumber;
    }

    public int getTotalNumber() {
        return totalNumber;
    }

    public void setTotalNumber(int totalNumber) {
        this.totalNumber = totalNumber;
        this.count();
    }

    public int getCurrentPage() {
        return currentPage;
    }

    public void setCurrentPage(int currentPage) {
        this.currentPage = currentPage;
    }

    public int getTotalPage() {
        return totalPage;
    }

    public void setTotalPage(int totalPage) {
        this.totalPage = totalPage;
    }

    public int getPageNumber() {
        return pageNumber;
    }

    public void setPageNumber(int pageNumber) {
        this.pageNumber = pageNumber;
        this.count();
    }

    public int getDbIndex() {
        return dbIndex;
    }

    public void setDbIndex(int dbIndex) {
        this.dbIndex = dbIndex;
    }

    public int getDbNumber() {
        return dbNumber;
    }

    public void setDbNumber(int dbNumber) {
        this.dbNumber = dbNumber;
    }
}

二、拦截器

package com.ming.interceptor;

import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.util.Map;
import java.util.Properties;

import org.apache.ibatis.executor.parameter.ParameterHandler;
import org.apache.ibatis.executor.statement.StatementHandler;
import org.apache.ibatis.mapping.BoundSql;
import org.apache.ibatis.mapping.MappedStatement;
import org.apache.ibatis.plugin.Interceptor;
import org.apache.ibatis.plugin.Intercepts;
import org.apache.ibatis.plugin.Invocation;
import org.apache.ibatis.plugin.Plugin;
import org.apache.ibatis.plugin.Signature;
import org.apache.ibatis.reflection.MetaObject;
import org.apache.ibatis.reflection.SystemMetaObject;

import com.ming.util.Page;

/**
 * 
 * 拦截器签名:要拦截的目标类型是StatementHandler(注意:type只能配置成接口类型),
 * 拦截的方法是名称为prepare参数为Connection类型的方法
 *
 */
@Intercepts({ @Signature(type = StatementHandler.class, method = "prepare", args = { Connection.class }) })
public class PageInterceptor1 implements Interceptor {

    public Object intercept(Invocation invocation) throws Throwable {
        // 拦截获取statement处理类
        StatementHandler statementHandler = (StatementHandler) invocation.getTarget();
        // 把statement处理类交给MetaObject代理类代理
        MetaObject metaObject = MetaObject.forObject(statementHandler, SystemMetaObject.DEFAULT_OBJECT_FACTORY,
                SystemMetaObject.DEFAULT_OBJECT_WRAPPER_FACTORY);
        MappedStatement mappedStatement = (MappedStatement) metaObject.getValue("delegate.mappedStatement");
        String id = mappedStatement.getId();
        //以ByPage结尾的才操作
        if (id.matches(".+ByPage$")) {
            BoundSql boundSql = statementHandler.getBoundSql();
            String sql = boundSql.getSql();
            String countSql = "select count(*) from (" + sql + ")a";
            // 获取connection
            Connection connection = (Connection) invocation.getArgs()[0];
            // 通过connection获取prepareStatement
            PreparedStatement countStatement = connection.prepareStatement(countSql);
            // 获取参数处理类
            ParameterHandler parameterHandler = (ParameterHandler) metaObject.getValue("delegate.parameterHandler");
            // 通过参数处理类 把参数设置进去
            parameterHandler.setParameters(countStatement);
            // 开始查询
            ResultSet rs = countStatement.executeQuery();
            // 获取传过来的参数
            Map<?, ?> parameter = (Map<?, ?>) boundSql.getParameterObject();
            // 获取page参数
            Page page = (Page) parameter.get("page");
            if (rs.next()) {
                // 为page设置查询出来的所有数据条数
                page.setTotalNumber(rs.getInt(1));
            }
            // 改造后带分页查询的SQL语句
            String pageSql = sql + " limit " + page.getDbIndex() + "," + page.getDbNumber();
            // 代理类设置改装好sql语句
            metaObject.setValue("delegate.boundSql.sql", pageSql);
        }
        return invocation.proceed();
    }

    public Object plugin(Object target) {
        //不合条件的 直接放行
        return Plugin.wrap(target, this);
    }

    public void setProperties(Properties properties) {
    }

}

小例子下载地址:

Mybatis拦截器分页是指通过使用Mybatis拦截器机制来实现数据库查询结果的分页显示。拦截器的作用是在方法执行前后加入自定义的逻辑代码,从而实现对方法的拦截和处理。在Mybatis中,我们可以使用拦截器来拦截Executor接口中的query方法,该方法用于执行数据库查询操作。通过拦截query方法,我们可以在查询前后进行一些处理,比如对查询结果进行分页处理。 在实现Mybatis拦截器分页的过程中,我们需要定义一个实现了Interceptor接口的拦截器类,并在该类上使用@Intercepts注解来定义拦截点。在拦截器类中,我们可以通过重写intercept方法来实现对query方法的拦截和处理。在intercept方法中,我们可以获取到方法的参数和执行结果,并根据需要进行分页处理。 为了在Mybatis中使用拦截器分页,我们还需要在Mybatis的配置文件中注册拦截器。在配置文件中,我们可以使用<plugins>标签来注册拦截器,指定拦截器的类名。通过注册拦截器,我们可以使拦截器生效,并在查询操作中进行分页处理。 需要注意的是,不同的数据库可能有不同的分页语法,为了保证程序的兼容性,最好将数据库方言的设置抽出来。可以定义一个数据库方言配置接口,通过实现该接口来设置不同数据库的分页语法。在分页处理中,我们可以根据数据库方言来生成相应的分页SQL语句,从而实现对不同数据库的兼容性支持。 #### 引用[.reference_title] - *1* *2* *3* [MyBatis拦截器实现分页](https://blog.youkuaiyun.com/wqh8522/article/details/78972135)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值