网站首页 > 文章精选 正文
一,概述
mybatis的分页使用的关键字就是limit,插件分页的原理就是在sql语句中拼接limit关键字,进行数据的分页查询,所以pageHelper也不例外,底层处理的就是使用的代理对象拼接的sql,实现的分页。其实思想都离不开原始的分页的sql语句,虽然这样的分页插件好用,但是建议初学者最好不要使用,如果要使用,也要吃透底层的原理才可以加深理解, 否则的话,可能只会使用不能明白其中的原理。
pageHelper使用的是threadLocal本地线程变量的特性来进行存储分页的的部分数据,使用到分页查询的参数组装,而且在使用一次后,就会被remove,但是这个是必须的,一个是防止分页数据被其他sql误用,一个是防止内存泄露。使用的是mybatis的实现Interceptor接口,使用jdk动态代理的模式,增强Executor对象,实现的代理对象,组装count和limit分页数据的查询的sql语句,返回一个page对象,page对象其实就是继承了list对象,所以返回的List对象是父类,适应多态。
二,原理分析
设置分页数据
PageHelper.startPage(重写startPage)
封装分页的pageNum pageSize以及orderFiled的一些数据
public static <E> Page<E> startPage(int pageNum, int pageSize, String orderBy) {
Page<E> page = startPage(pageNum, pageSize);
page.setOrderBy(orderBy);
return page;
}
public static <E> Page<E> startPage(int pageNum, int pageSize, boolean count, Boolean reasonable, Boolean pageSizeZero) {
Page<E> page = new Page(pageNum, pageSize, count);
page.setReasonable(reasonable);
page.setPageSizeZero(pageSizeZero);
Page<E> oldPage = getLocalPage();
if (oldPage != null && oldPage.isOrderByOnly()) {
page.setOrderBy(oldPage.getOrderBy());
}
//设置分页的数据存放在ThreadLocal里面,
setLocalPage(page);
return page;
}
以上是可以看出,分页的数据是和ThreadLocal绑定的,也是将数据存放在ThreadLocal里面,提供给后续分页的时候使用
清除ThreadLocal分页数据
以上图你可以看出,清除数据的是在
com.github.pagehelper.PageInterceptor#intercept方法中的finally中执行的,概述也说过,其实执行的原理就是利用的mybatis的intercept的进行的对象的代理
PageInterceptor代理的源头
@Intercepts注解就是标注需要拦截代理的对象,type就是对象为Executor,方法是query,参数为两个重载的方法,PageInterceptor实现Interceptor并实现intercept和plugin方法,进行数据代理的处理,setProperties方法其实是可以设置配置参数来处理,实现接口原因是因为mybatis使用的jdk动态代理,必须是实现接口的才可以创建代理对象,下面就intercept方法进行分析分页的处理逻辑
intercept方法
this.count进行查询统计总数
就是使用原sql语句进行count查询总数
com.github.pagehelper.util.ExecutorUtil#pageQuery
dialect.getPageSql就是拼接sql语句中的limit
看到这里,可能就可以明白了,这个分页到底是怎么做的 了,其实就是sql拼接了limit,总上图就可以看出,规归根结底就是代理了Excutor对象,然后执行count和拼接limit的sql语句,实现的插件的分页
小结
以上可以看出来,其实就是一个Excutor代理对象的一系列处理,简单而言,pageHelper的分页逻辑处理,可以就这么理解,其实就是生成代理对象并代理增强,从ThreadLocal中获取到分页的参数,拼接sql和查询总数的sql并返回一个page对象
三,Excutor逻辑补充
首先写分页插件的话,需要知道mybatis的原理,执行Sql的是四大对象:Executor,StatementHandler,ParameterHandler,ResultSetHandler。
一般都会获取SqlSession,获取途径一般都是SqlSessionFactory .openSession
Executor的创建
增强对象
从上面可以看出,其实Executor创建的过程就是执行了拦截器inteceptor中的拦截器链
猜你喜欢
- 2025-06-09 PageHelper - 最方便的 MyBatis 分页插件
- 2025-06-09 50个Java编程技巧,免费送给大家(java编程基础知识入门)
- 2025-06-09 SpringBoot 各种分页查询方式详解(全网最全)
- 2025-06-09 SpringBatch - R&W, 我与富婆的这一年
- 2025-06-09 面试官:说说MyBatis分页插件(PageHelper)工作原理和配置过程?
- 2025-06-09 面试二:pagehelper是怎么实现分页的,
- 2025-06-09 SpringBoot集成Mybatis-Plus分页插件
- 2025-06-09 【开发技术】Mybatis中进行多表关联查询?性能是不是会变好呢?
- 2025-06-09 每天从外包系统同步百万数据,用什么方案?Java实战讲解
- 2025-06-09 MyBatis插件开发实战:手写一个分页插件
- 最近发表
-
- 面试中常被问到的Hash表,你了解吗
- JAVA面试考点:一文搞懂一致性Hash的原理和实现
- 一次性搞清楚equals和hashCode(hashcode() 与equals()区别,简单说明)
- HashMap.Key的故事:Key为什么出现Hash碰撞及冲突呢?
- hash冲突的几种解决方案对比(hash冲突的解决方式)
- 游戏王LN 无头骑士(无头骑士cv)
- Linux ln、unlink命令用法(linux link命令详解)
- n和l分不清矫正发音方法,这三步就够了
- golang引用私有gitlab项目代码(golang引入当前包下的文件)
- Instamic:录音领域中的 GoPro,让你想录就录,随心所欲
- 标签列表
-
- newcoder (56)
- 字符串的长度是指 (45)
- drawcontours()参数说明 (60)
- unsignedshortint (59)
- postman并发请求 (47)
- python列表删除 (50)
- 左程云什么水平 (56)
- 计算机网络的拓扑结构是指() (45)
- 编程题 (64)
- postgresql默认端口 (66)
- 数据库的概念模型独立于 (48)
- 产生系统死锁的原因可能是由于 (51)
- 数据库中只存放视图的 (62)
- 在vi中退出不保存的命令是 (53)
- 哪个命令可以将普通用户转换成超级用户 (49)
- noscript标签的作用 (48)
- 联合利华网申 (49)
- swagger和postman (46)
- 结构化程序设计主要强调 (53)
- 172.1 (57)
- apipostwebsocket (47)
- 唯品会后台 (61)
- 简历助手 (56)
- offshow (61)
- mysql数据库面试题 (57)