最近看了一下项目中代码,发现系统中使用的mybatis分页使用的是mybatis自带的分页,即使用RowBounds来进行分页,而这种分页是基于内存分页,即一次查出所有的数据,然后再返回分页需要的数据。断点跟踪mybatis的源码可以看到是这个方法中org.apache.ibatis.executor.resultset.FastResultSetHandler.skipRows(ResultSet, RowBounds)决定是将resultset的游标从哪个地方开始返回数据.

如果我们在mybatis执行sql之前,动态的将普通的sql语句换成分页的sql即可解决问题。那么我们就要找到mybatis是合适进行预编译sql的。

通过在程序中打断点,我们知道先是MapperProxy进行处理->然后是创建MapperMethod->然后执行mapperMethod的execute方法,在这个方法中判断我们是那种执行操作,比如增加,修改等,假如是查询操作(SELECT),然后就会根据mapperMethod中的sqlSessionTemplete参数,进行查询(selectList)操作,在该方法中进行SqlSession的获取,获取sqlSession时,会创建Exector对象(默认是SimpleExecutor)最终返回DefaultSqlSession, 然后经过一系列的操作,执行DefaultSqlSession中的selectList方法,在该方法中会执行executor的query方法,再经过一系列的操作,会执行org.apache.ibatis.executor.SimpleExecutor.doQuery(MappedStatement, Object, RowBounds, ResultHandler, BoundSql)这个方法,在该方法中会创建StatementHandler对象,跟踪断点知道创建的是RoutingStatementHandler,在RoutingStatementHandler的构造方法中,会根据StatementType判断创建那种Statement,默认是创建PreparedStatementHandler,在RoutingStatementHandler创建成功后,就会执行RoutingStatementHandler的prepare方法,在该方法中就会执行connection.prepareStatement(sql);最终返回java.sql.Statement对象。那么我们只要在connection.prepareStatement(sql)执行之前将不是分页的sql替换成分页的sql即可实现数据库的分页。

(上述流程建议,打一个断点跟踪一下mybatis的执行过程。)

     经过上述分析,我们知道sql语句是由org.apache.ibatis.executor.statement.StatementHandler.prepare(Connection)这个方法来执行预编译的。那么我们就可以写一个拦截器拦截prepare方法,在sql语句预编译之前改变sql,这正好可以借助mybatis的插件来实现。

在mybatis中编写一个插件需要实现org.apache.ibatis.plugin.Interceptor这个接口,然后在该接口上使用注解(@Intercepts)来标注要拦截的方法

解释一下插件中可能要用到的几个类:

MetaObject:mybatis提供的一个基于返回获取属性值的对象的类

    BoundSql : 在这个里面可以获取都要执行的sql和执行sql要用到的参数

    MappedStatement : 这个可以得到当前执行的sql语句在xml文件中配置的id的值

    RowBounds : 是mybatis内存分页要用到的。

    ParameterHandler : 是mybatis中用来替换sql中?出现的值的.

    那么我就说一下,我在我们项目中实现分页的实现:

    1.基于oracel分页

2.拦截select操作,其余的操作放行,即根据mappedStatement.getId()方法来判断

3.对分页中的参数进行一个判断,只要参数的类型是map,并且存在pageNo和pageSize这2个属性,即任务是分页sql(我们系统中的参数都是以map的方式传递的)

4.计算出总的记录数,然后放回到传递参数的map中

@Intercepts({ @Signature(type = StatementHandler.class, method = "prepare", args = { Connection.class }) })
public class PageInterceptor implements Interceptor { @Override
public Object intercept(Invocation invocation) throws Throwable {
StatementHandler statementHandler = (StatementHandler) invocation.getTarget();
MetaObject metaStatementHandler = MetaObject.forObject(statementHandler, SystemMetaObject.DEFAULT_OBJECT_FACTORY, SystemMetaObject.DEFAULT_OBJECT_WRAPPER_FACTORY);
BoundSql boundSql = statementHandler.getBoundSql();
MappedStatement mappedStatement = (MappedStatement) metaStatementHandler.getValue("delegate.mappedStatement");
String id = mappedStatement.getId();
System.out.println("sql语句的id : " + id);
Object param = boundSql.getParameterObject();
if (null != param && Map.class.isAssignableFrom(param.getClass())) {
Map paramMap = (Map) param;
Object _pageSize = paramMap.get("pageSize");
Object _pageNo = paramMap.get("pageNo");
if (_pageNo != null && _pageSize != null) {
int pageSize = (Integer) _pageSize;
int pageNo = (Integer) _pageNo;
setTotalResult(boundSql, (Connection) invocation.getArgs()[0], metaStatementHandler, paramMap);
String sql = getPageSql(pageSize, pageNo, boundSql);
metaStatementHandler.setValue("delegate.boundSql.sql", sql);
metaStatementHandler.setValue("delegate.rowBounds.offset", RowBounds.NO_ROW_OFFSET);
metaStatementHandler.setValue("delegate.rowBounds.limit", RowBounds.NO_ROW_LIMIT);
}
}
return invocation.proceed();
} // 这个分页方法是借助于网上的
private String getPageSql(int pageSize, int pageNo, BoundSql boundSql) {
StringBuffer sql = new StringBuffer(boundSql.getSql());
sql.insert(0, "select u.*, rownum r from (").append(") u where rownum < ").append(pageSize * pageNo);
sql.insert(0, "select * from (").append(") where r >= ").append((pageSize - 1) * pageNo);
return sql.toString();
}
private void setTotalResult(BoundSql boundSql, Connection conn, MetaObject metaObject, Map param) throws SQLException {
String countSql = "select count(*) from ( " + boundSql.getSql() + " ) total";
PreparedStatement prepareStatement = conn.prepareStatement(countSql);
ParameterHandler parameterHandler = (ParameterHandler) metaObject.getValue("delegate.parameterHandler");
parameterHandler.setParameters(prepareStatement); // 给sql语句设置参数
ResultSet resultSet = prepareStatement.executeQuery();
if (resultSet.next()) {
param.put("total", resultSet.getObject(1));
}
if (resultSet != null) {
resultSet.close();
}
if (prepareStatement != null) {
prepareStatement.close();
}
} @Override
public Object plugin(Object target) {
return Plugin.wrap(target, this);
}
@Override
public void setProperties(Properties properties) { }
}

插件写好之后,在配置文件中进行配置:

<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSource" />
<property name="mapperLocations" value="sqlmap/*.xml"></property>
<property name="plugins">
<array>
<bean class="com.study.mybatis.spring.interceptor.PageInterceptor" /> // 插件所在的类中的全路径
</array>
</property>
</bean>

mybatis自定义分页拦截器的更多相关文章

  1. Mybatis自定义SQL拦截器

    本博客介绍的是继承Mybatis提供的Interface接口,自定义拦截器,然后将项目中的sql拦截一下,打印到控制台. 先自定义一个拦截器 package com.muses.taoshop.com ...

  2. 【mybatis】mybatis分页拦截器搭配bootstrap-table使用

    提前说明: 这一种方式已被我自己pass掉了,已经被新的方式迭代了.但是记录下自己曾经的成果还是有必要的,而且里面的思想还是不变的,另外技术不就是在不断地迭代中升级吗.千万不要想着一步完美,那样会让你 ...

  3. Mybatis中的拦截器

    作者:moshenglv的专栏 拦截器的一个作用就是我们可以拦截某些方法的调用,我们可以选择在这些被拦截的方法执行前后加上某些逻辑,也可以在执行这些被拦截的方法时执行自己的逻辑而不再执行被拦截的方法. ...

  4. SpringMVC 自定义一个拦截器

    自定义一个拦截器方法,实现HandlerInterceptor方法 public class FirstInterceptor implements HandlerInterceptor{ /** * ...

  5. Spring自定义一个拦截器类SomeInterceptor,实现HandlerInterceptor接口及其方法的实例

    利用Spring的拦截器可以在处理器Controller方法执行前和后增加逻辑代码,了解拦截器中preHandle.postHandle和afterCompletion方法执行时机. 自定义一个拦截器 ...

  6. Dubbo自定义日志拦截器

    前言 上一篇文章 Spring aop+自定义注解统一记录用户行为日志 记录了 web层中通过自定义注解配合Spring aop自动记录用户行为日志的过程.那么按照分布式架构中Dubbo服务层的调用过 ...

  7. springmvc自定义的拦截器以及拦截器的配置

    一.自定义拦截器 Spring MVC也可以使用拦截器对请求进行拦截处理,用户可以自定义拦截器来实现特定的功能,自定义的拦截器必须实现HandlerInterceptor接口. 二.HandlerIn ...

  8. MyBatis功能点二:MyBatis提供的拦截器平台

    前面关于MyBatis功能点二plugin已经介绍了一些应用及其实现的底层代码,本文总结MyBatis提供的拦截器平台框架体系. 通过MyBatis功能点二:从责任链设计模式的角度理解插件实现技术 - ...

  9. MyBatis空where拦截器

    最近项目中出现了至少两次因为Mybatis的动态where条件不满足导致实际sql语句的where条件为空,进而查询全表,当数据量比较大的时候,导致OOM的情况. 如何禁止这种情况,个人觉得三种措施: ...

随机推荐

  1. Intel® QAT 加速卡之IPSec示例

    Intel QAT 加速卡之IPSec示例 文章目录 Intel QAT 加速卡之IPSec示例 1. QAT处理IPSec入站报文 2. QAT处理IPSec出站报文 3. 组织架构 4. 示例源码 ...

  2. 只需3步,快来用AI预测你爱的球队下一场能赢吗?

    摘要:作为球迷,我们有时候希望自己拥有预测未来的能力. 本文分享自华为云社区<用 AI 预测球赛结果只需三步,看看你爱的球队下一场能赢吗?>,作者:HWCloudAI. 还记得今年夏天的欧 ...

  3. python库--requests

    requests 方法 返回 参数 方法详情 .get()  r  url  get请求 params  url?后面的内容会以'key=value'的方式接到url后面 proxies 设置代理ip ...

  4. MongoDB索引的简单理解

    目录 MongoDB索引 1.语法准备 2.数据准备: 3.索引 3.1 唯一索引 3.2 单键索引 3.3 多键索引 3.4 复合索引 3.5 交叉索引 3.6 部分索引 3.7覆盖索引 3.8 全 ...

  5. ❤️❤️用最简单的方法在Webstorm中打开已存在项目 和 新建Vue项目 (亲测实用)❤️❤️

    ​ 目录 一:打开已存在项目时 二:新建一个vue项目 使用webstorm创建vue项目创建vue项目各个公司用的工具都不一样 最常见的有HBuilder X,WebStorm,Visual Stu ...

  6. centos7 下安装 mysql5.7

    由于CentOS7的yum源中没有mysql,需要到mysql的官网下载yum repo配置文件. 下载命令: wget https://dev.mysql.com/get/mysql57-commu ...

  7. JetBrains 系列软件汉化包 2017.3-2018.1

    JetBrains 系列软件汉化包 关键字: Android Studio 3.0-3.1.3 汉化包 CLion 2018.1-2018.2 汉化包 GoLand 2017.3.2-2018.2 汉 ...

  8. 关于goto

    (下面一段来源<征服C指针>) 75: ReadLineStatus read_line(FILE *fp, char **line) 76: { 77: int ch; 78: Read ...

  9. Mysql backup and Recovery Data Type.

    数据库备份方法: 备份类型:物理备份和逻辑备份: 物理备份是指直接复制存储数据库内容的目录和文件,这种类型的备份适用于出现问题时需要快速恢复的大型重要数据库.逻辑备份保存以逻辑数据库结构(create ...

  10. 支付宝openssl_sign(): supplied key param cannot be coerced into a private key in

    先说一下,生成rsa 私钥 公钥的方法,以ubuntu 为例sudo apt-get install openssl # 先装上这个库genrsa -out rsa_private_key.pem 1 ...