Spring Mybatis 分页插件使用教程

Mybatis分页切入点

Mybatis内部有个plugins(插件)概念,本质上属于拦截器的思想。具体的解析可见他文MyBatis拦截器原理探究。本文将在此基础上直接展示实际项目的实现代码和其他的相关解析

分页具体代码实现

首先我们可以定义方言抽象类,用于实现分页AbstractDialect.java

public abstract class AbstractDialect{
 /**
  * 是否支持limit和偏移量
  * @return
  */
 public abstract boolean supportsLimitOffset();
 /**
  * 是否支持limit
  * @return
  */
 public abstract boolean supportsLimit();
 /**
  * 获取增加了分页属性之后的SQL
  * @param sql
  * @param offset
  * @param limit
  * @return
  */
 public abstract String getLimitString(String sql, int offset, int limit);
}

再而我们就以Oracle与Mysql数据库的分页技术作下分别的实现

MySQLDialect.java-Mysql分页方言

public class MySQLDialect extends AbstractDialect {
 public boolean supportsLimitOffset() {
  return true;
 }
 public boolean supportsLimit() {
  return true;
 }
 public String getLimitString(String sql, int offset, int limit) {
  if (offset > 0) {
   return sql + " limit " + offset + "," + limit;
  } else {
   return sql + " limit " + limit;
  }
 }
}

OracleDialect.java-Oracle方言实现

public class OracleDialect extends ADialect{
 @Override
 public boolean supportsLimitOffset() {
  return false;
 }
 @Override
 public boolean supportsLimit() {
  return false;
 }
 @Override
 public String getLimitString(String sql, int start, int limit) {
  if(start < 0){
   start = 0;
  }
  if(limit < 0){
   limit = 10;
  }
  StringBuilder pageSql = new StringBuilder(100);
  pageSql.append("select * from ( select temp.*, rownum row_id from ( ");
  pageSql.append(sql);
  pageSql.append(" ) temp where rownum <= ").append(start+limit);
  pageSql.append(") where row_id > ").append(start);
  return pageSql.toString();
 }
}

对应的Mybatis插件拦截器实现如下,拦截StatementHandler#prepare(Connection con)创建SQL语句对象方法

PaginationInterceptor.java

@Intercepts({ @Signature(type = StatementHandler.class, method = "prepare", args = { Connection.class }) })
public final class PaginationInterceptor implements Interceptor {
 private final static Logger log = LoggerFactory
  .getLogger(PaginationInterceptor.class);
 private ADialect dialect;
 public void setDialect(ADialect dialect) {
  this.dialect = dialect;
 }
 @Override
 public Object intercept(Invocation invocation) throws Throwable {
  // 直接获取拦截的对象,其实现类RoutingStatementHandler
  StatementHandler statementHandler = (StatementHandler) invocation
   .getTarget();
  BoundSql boundSql = statementHandler.getBoundSql();
  // 获取元对象,主要用于获取statementHandler所关联的对象及属性
  MetaObject metaStatementHandler = MetaObject.forObject(
   statementHandler, new DefaultObjectFactory(),
   new DefaultObjectWrapperFactory());
  MappedStatement mappedStmt= (MappedStatement) metaStatementHandler
   .getValue("delegate.mappedStatement".intern());
  // 只对queryPagination()方法进行分页操作
  if(mappedStmt.getId().indexOf("queryPagination")==-1){
   return invocation.proceed();
  }
  // 重新构造分页的sql
  String originalSql = (String) metaStatementHandler
   .getValue("delegate.boundSql.sql".intern());
  metaStatementHandler.setValue("delegate.boundSql.sql".intern(), dialect
   .getLimitString(originalSql, rowBounds.getOffset(),
    rowBounds.getLimit()));
  metaStatementHandler.setValue("delegate.rowBounds.offset".intern(),
   RowBounds.NO_ROW_OFFSET);
  metaStatementHandler.setValue("delegate.rowBounds.limit".intern(),
   RowBounds.NO_ROW_LIMIT);
  log.debug("page sql : " + boundSql.getSql());
  return invocation.proceed();
 }
 // 拦截对象
 @Override
 public Object plugin(Object target) {
  return Plugin.wrap(target, this);
 }
 @Override
 public void setProperties(Properties properties) {
 }
}

Spring对应的xml配置可如下,以oracle分页为例子

<!-- oracle方言配置,用于oracle的分页 -->
 <bean id="paginationInterceptor"  class="com.hsnet.winner.cas.admin.core.dao.mybatis.interceptor.PaginationInterceptor">
  <property name="dialect">
   <bean class="cn.cloud.winner.oss.manager.mybatis.page.OracleDialect" />
  </property>
 </bean>

使用以上的代码以及配置即可完成对oracle数据库以及mysql数据库的分页操作。并且博主对其中的某个点作下解析

Mybatis#MetaObject-元数据对象解析

以上的代码博主之前在使用的时候,对其中的MetaObject这个类很费解,其直接通过getValue()方法便可以将所代理的对象的所关联的属性全都拿取到。我们可以跟随源码深入了解下

MetaObject#forObject()

代理对象均通过此静态方法进入

public static MetaObject forObject(Object object, ObjectFactory objectFactory, ObjectWrapperFactory objectWrapperFactory) {
 if (object == null) {
  return SystemMetaObject.NULL_META_OBJECT;
 } else {
  return new MetaObject(object, objectFactory, objectWrapperFactory);
 }
 }

我们可以直接观察其中的构造函数,玄机就在此处

private MetaObject(Object object, ObjectFactory objectFactory, ObjectWrapperFactory objectWrapperFactory) {
 this.originalObject = object;
 this.objectFactory = objectFactory;
 this.objectWrapperFactory = objectWrapperFactory;
 // 所有的属性获取均通过objectWrapper类来获取,此处主要对所代理的object对象类型进行判断
 if (object instanceof ObjectWrapper) {
  this.objectWrapper = (ObjectWrapper) object;
 } else if (objectWrapperFactory.hasWrapperFor(object)) {
  this.objectWrapper = objectWrapperFactory.getWrapperFor(this, object);
 } else if (object instanceof Map) {
  this.objectWrapper = new MapWrapper(this, (Map) object);
 } else if (object instanceof Collection) {
  this.objectWrapper = new CollectionWrapper(this, (Collection) object);
 } else {
  // 我们常用的便是BeanWrapper
  this.objectWrapper = new BeanWrapper(this, object);
 }
 }

为了理解的更为渗透,我们继续跟进,最后我们得知其会调用Reflector类的构造函数

private Reflector(Class<?> clazz) {
 type = clazz;
 // 获取构造类
 addDefaultConstructor(clazz);
 // 获取get方法集合
 addGetMethods(clazz);
 // 获取set方法集合
 addSetMethods(clazz);
 // 获取内部属性集合
 addFields(clazz);
 readablePropertyNames = getMethods.keySet().toArray(new String[getMethods.keySet().size()]);
 writeablePropertyNames = setMethods.keySet().toArray(new String[setMethods.keySet().size()]);
 for (String propName : readablePropertyNames) {
  caseInsensitivePropertyMap.put(propName.toUpperCase(Locale.ENGLISH), propName);
 }
 for (String propName : writeablePropertyNames) {
  caseInsensitivePropertyMap.put(propName.toUpperCase(Locale.ENGLISH), propName);
 }
 }

由此我们便可知使用Reflector代理类以及MetaObject便可以遍历代理被代理类的所关联的所有属性,就拿RoutingStatementHandler类来说,经过上述操作后其便可以访问内部属性delegate以及delegate的内部属性configuration/objectFactory/typeHandlerRegistry/resultSetHandler/parameterHandler/mappedStatement等属性

MetaObject#getValue()

上述阐述的是如何代理被代理类的内部属性,我们也简单的看下是如何正确的调用

public Object getValue(String name) {
 // PropertyTokenizer与StringTokenizer类似,只是前者写死以.为分隔符
 PropertyTokenizer prop = new PropertyTokenizer(name);
 if (prop.hasNext()) {
  MetaObject metaValue = metaObjectForProperty(prop.getIndexedName());
  if (metaValue == SystemMetaObject.NULL_META_OBJECT) {
  return null;
  } else {
  return metaValue.getValue(prop.getChildren());
  }
 } else {
  return objectWrapper.get(prop);
 }
 }

具体的解析就不在此阐述了,如何用户想获取StatementHandler所拥有的sql字符串,可通过getValue("delegate.boundSql.sql")其中以.为分隔符并其中的属性必须是内部属性(区分大小写)。

MetaObject#setValue()

原理同MetaObject#getValue()方法

总结

以上所述是小编给大家介绍的Spring Mybatis 分页插件使用教程,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

您可能感兴趣的文章:

  • Spring MVC+MyBatis+MySQL实现分页功能实例
  • SpringMvc+Mybatis+Pagehelper分页详解
  • Java简单实现SpringMVC+MyBatis分页插件
(0)

相关推荐

  • Spring MVC+MyBatis+MySQL实现分页功能实例

    前言 最近因为工作的原因,在使用SSM框架实现一个商品信息展示的功能,商品的数据较多,不免用到分页,查了一番MyBatis分页的做法,终于是实现了,在这里记录下来分享给大家,下面来一起看看详细的介绍: 方法如下:  首先写一个分页的工具类,定义当前页数,总页数,每页显示多少等属性. /** * 分页 工具类 */ public class Page implements Serializable { private static final long serialVersionUID = -22

  • SpringMvc+Mybatis+Pagehelper分页详解

    最近公司需要做一个告警页面的功能,需要分页,查了很多资料发现PageHelper比较合适 故写一篇从零开始的PageHelper使用的教程,也记录下忙活一天的东西 1.首先需要在项目中添加PageHelper的依赖,这里我用的Maven添加 <dependency> <groupId>com.github.pagehelper</groupId> <artifactId>pagehelper</artifactId> <version>

  • Java简单实现SpringMVC+MyBatis分页插件

    1.封装分页Page类 package com.framework.common.page.impl; import java.io.Serializable; import com.framework.common.page.IPage; /** * * * */ public abstract class BasePage implements IPage, Serializable { /** * */ private static final long serialVersionUID

  • Spring Mybatis 分页插件使用教程

    Mybatis分页切入点 Mybatis内部有个plugins(插件)概念,本质上属于拦截器的思想.具体的解析可见他文MyBatis拦截器原理探究.本文将在此基础上直接展示实际项目的实现代码和其他的相关解析 分页具体代码实现 首先我们可以定义方言抽象类,用于实现分页AbstractDialect.java public abstract class AbstractDialect{ /** * 是否支持limit和偏移量 * @return */ public abstract boolean

  • Mybatis分页插件的实例详解

    Mybatis分页插件的实例详解 1.前言: 我们知道,在MySQL中,分页的sql是使用limit来做,如果我们自己写sql,那分页肯定是没有任何问题的.但是一旦model多了起来,复杂了起来,我们很自然的想到使用mybatis的逆向工程来生成相应的po和mapper,但是同时也会带来弊端,比如这里的分页问题就不好解决了. 可能有人会说,我可以修改生成的文件,没错,这是可行的,但是一般我们通过逆向工程生成的文件,都不会去动它,所以这个时候,就需要使用分页插件来解决了. 如果你也在用Mybati

  • mybatis分页插件pageHelper详解及简单实例

    mybatis分页插件pageHelper详解及简单实例 工作的框架spring springmvc mybatis3 首先使用分页插件必须先引入maven依赖,在pom.xml中添加如下 <!-- 分页助手 --> <dependency> <groupId>com.github.pagehelper</groupId> <artifactId>pagehelper</artifactId> <version>3.7.5

  • Mybatis分页插件PageHelper的配置和简单使用方法(推荐)

    前言 在web开发过程中涉及到表格时,例如dataTable,就会产生分页的需求,通常我们将分页方式分为两种:前端分页和后端分页. 前端分页 一次性请求数据表格中的所有记录(ajax),然后在前端缓存并且计算count和分页逻辑,一般前端组件(例如dataTable)会提供分页动作. 特点是:简单,很适合小规模的web平台:当数据量大的时候会产生性能问题,在查询和网络传输的时间会很长. 后端分页 在ajax请求中指定页码(pageNum)和每页的大小(pageSize),后端查询出当页的数据返回

  • Mybatis分页插件使用方法详解

    本文实例为大家分享了Mybatis分页插件使用的具体代码,供大家参考,具体内容如下 1.分页插件简介 pagehelper源码 都说这是史上最好用的分页插件,支持多种数据库以多种方式分页. 2.分页插件的使用 2.1导入maven依赖 <dependency> <groupId>com.github.pagehelper</groupId> <artifactId>pagehelper</artifactId> <version>最新

  • SSM使用mybatis分页插件pagehepler实现分页示例

    前几天在这里分享了手写 sql 分页查询实现分页,现在来看看使用 mybatis 分页插件 pagehepler 来实现分页 使用分页插件的原因,简化了sql代码的写法,实现较好的物理分页,比写一段完整的分页sql代码,也能减少了误差性. Mybatis分页插件 demo 项目地址:free-Mybatis_PageHelper_jb51.rar 我这里使用 maven 工程实现: 1.首先导入分页插件的依赖: <dependency> <groupId>com.github.pa

  • MyBatis分页插件PageHelper的具体使用

    MyBatis分页插件PageHelper 如果你也在用 MyBatis,建议尝试该分页插件,这一定是最方便使用的分页插件.分页插件支持任何复杂的单表.多表分页. PageHelper是一个Mybatis的分页插件, 负责将已经写好的sql语句, 进行分页加工. PageHelper的使用 优点:无需你自己去封装以及关心sql分页等问题,使用很方便,前端取数据也很方便. 1.引入pagehelper依赖 <dependency> <groupId>com.github.pagehe

  • SpringBoot+Mybatis分页插件PageHelper实现分页效果

    目录 一.项目结构 二.插件引入 三.代码 四.测试: 最近刚入职新公司,项目是从零开始搭建的项目.我觉得是时候考验是驴还是千里马的时候.都是泪就不多说了. 附上一篇Mybatis常用的分页案例.这次要做的是最常见的分页效果,也是基础功能.但是很多人都做不好的.这次采用Mybatis分页插件PageHelper.   仅献给伸手党的大爷们.大爷们好!拿代码记得扣666!!小的在这给感谢了!! 一.项目结构 按照controller,service,mapper(也叫dao)来建立项目,utils

  • Mybatis分页插件PageHelper手写实现示例

    目录 引言 编写我们的插件类 上面有二个核心方法 获取记录总数 分页查询记录数 如何获取前端传递过来的参数? 总结 引言 PageHelper是一个非常好用的插件,以至于很想知道它底层是怎么实现的.至于MyBatis插件概念原理网上有很多,我不太喜欢去写一些概念性的东西,我比较喜欢自己动手实现的那种,话不多说,我们开干 搭建一个SpringBoot+MyBatis+MySql项目 编写我们的插件类 package com.example.demo.plugin; import org.apach

  • 解决mybatis分页插件PageHelper导致自定义拦截器失效

    目录 问题背景 mybatis拦截器使用 使用方法: 注解参数介绍: setProperties方法 bug内容: 自定义拦截器部分代码 PageInterceptor源码: 解决方法: 解决方案一 调整执行顺序 解决方案二 修改拦截器注解定义 问题背景 在最近的项目开发中遇到一个需求 需要对mysql做一些慢查询.大结果集等异常指标进行收集监控,从运维角度并没有对mysql进行统一的指标搜集,所以需要通过代码层面对指标进行收集,我采用的方法是通过mybatis的Interceptor拦截器进行

随机推荐