PageHelper開源框架解讀

????????在使用springboot開發系統時,列表查詢經常會用PageHelper來進行分頁。使用起來很方便,但從未想過它的實現原理,所以對其進行解讀。

@Service
public class ScUserServiceImpl extends ServiceImpl<ScUserMapper, ScUser> implements IScUserService {@Overridepublic PageInfo<ScUser> pageList(UserListF form) {LambdaQueryWrapper<ScUser> qw = Wrappers.<ScUser>lambdaQuery().like(ScUser::getName, form.getName());// 開啟分頁,對下一條執行的sql進行分頁Page<ScUser> page = PageHelper.startPage(form.getPageNum(), form.getPageSize());List<ScUser> userList = baseMapper.selectList(qw);return new PageInfo<>(page);}
}

? ? ? ? 基于PageHelper(v5.3.2)官方文檔,這是一個最簡單、最常用的分頁demo,持久層框架使用的是Mybatis-plus.

? ? ? ? 調試截圖:

????????調試過程發現3個問題點:

  1. 為什么調用PageHelper.startPage()方法后可以實現分頁?
  2. 為什么在執行sql查詢前先執行select count(0) ?
  3. 為什么查詢出來的userList不需要return,直接return new PageInfo<>(page)就可以有數據?

? ? ? ? 追蹤過程:

? ? ? ? 進入PageHelper.startPage()方法,看到里面有一個 setLocalPage(page) 方法,方法的入參page里包含了分頁參數pageNum、pageSize等

????????setLocalPage方法對 LOCAL_PAGE 這個線程局部變量進行進行賦值

? ? ? ? 在getLocaPage方法打斷點,看什么時候會取出LOCAL_PAGE線程局部變量。最終發現方法落在了com.github.pagehelper.PageHelper#doBoundSql

????????同時在該方法的第二個入參boundSql里可以下一條要執行的sql

? ? ? ? 繼續追蹤,看到它的上級調用方法在com.github.pagehelper.PageInterceptor#intercept

?????????PageInterceptor這個類實現了ibatis的Interceptor接口,從而實現了對sql語句的攔截

????????intercept()這個方法的中文注釋非常完整,基本可以對照著閱讀源碼。

????????其中 dialect.beforeCount()這個方法會去線程局部變量LOCAL_PAGE中拿到count屬性,從而決定是否在分頁查詢前執行count(0)查詢。而這個count屬性在一開始我們調用PageHelper.startPage()的時候,就配置了默認值true。

? ? ? ? 而 ExecutorUtil.pageQuery() 則是對查詢sql進行解析(配置了多種解析器兼容mysql、oracle、sqlserver、Oscar等),并組裝成對應的分頁sql。組裝好的sql將其封裝為BoundSql對象,再調用org.apache.ibatis.executor.Executor#query()方法執行,最終拿到分頁查詢結果。? ? ??

? ? ? ? 根據現有的結論我們可以得出,分頁參數是通過ThreadLocal<Page>的方式傳遞的,那么第3個問題,為什么分頁查詢結果集不需要return而是直接return?page對象,答案也就水落石出了。

????????在分頁查詢結束后,dialect.afterPage()方法會將結果集resultList塞入到LOCAL_PAGE 這個線程局部變量里去。

? ? ? ? 問題點解答:

? ? ? ? 1、為什么調用PageHelper.startPage()方法后可以實現分頁?

? ? ? ? 因為startPage方法將分頁參數保存到了線程局部變量,然后通過PageInterceptor(它實現了ibatis提供的inteceptor接口)對下一條即將執行的query sql進行攔截,解析語義并組裝為分頁query sql,執行并拿到結果集,并存放到線程局部變量里去。

? ? ? ? 2、為什么在執行sql查詢前先執行select count(0) ?

? ? ? ? 因為調用?PageHelper.startPage(form.getPageNum(), form.getPageSize()) 方法時傳遞了默認count屬性值,默認為true,所以在分頁時會執行count(0)查詢總記錄數。它并不影響分頁的過程、結果,它只是滿足業務需要所做的一個查詢。

? ? ? ? 如果不需要count(0),可以調用?PageHelper.startPage(form.getPageNum(), form.getPageSize(), false) 來指定。

? ? ? ? 3、為什么查詢出來的userList不需要return,直接return new PageInfo<>(page)就可以有數據?

? ? ? ? 因為page是一個線程局部變量,分頁查詢結束后,PageInterceptor會調用 dialect.afterPage() 將結果集保存到page里去,所以page里會有數據。

? ? ? ? 你也可以return userList,但是page里的屬性更加豐富,它除了結果集之外,還有當前頁、每頁記錄數、總記錄數等等.....

? ? ? ? 小結:

? ? ? ? 經過代碼追蹤,我明白了PageHelper的本質其實是基于ThreadLocal和ibatis提供的sql監聽器實現的,解開了這層神秘的面紗。這種對sql約定好的的統一處理策略,非常值得學習。對也讓我對mybatis的的執行過程更加好奇,后續可以更進一步對mybatis的原理進行追蹤和理解。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/713592.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/713592.shtml
英文地址,請注明出處:http://en.pswp.cn/news/713592.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

WIN10 無密碼自動登錄

1、家里重裝了一下WIN10系統&#xff0c;第一次登陸居然用了微軟網站賬號&#xff0c;結果密碼忘記了&#xff0c;后面只能用PIN碼登陸系統。 2、需要登錄微軟的網站修改密碼&#xff1a; Microsoft account | Sign In or Create Your Account Today – Microsoft 3、在運行…

Linux-Prnt10:幾款國際打印機

這周接觸到惠普國際生產的幾款打印機設備&#xff0c;hplip的官網更新記錄里居然都沒有&#xff0c;特地確認了下其中緣由。這三款設備的型號分別是mpressora HP Laser 107w、mpressora Multifuncional HP Laser MFP 135a、mpressora Multifuncional HP Laser MFP 135w&#xf…

APP自動化測試-入門示例

入門示例 通過上一篇博客APP自動化測試介紹-CSDN博客的學習&#xff0c;相信大家對APP自動化測試已經有了一定的了解&#xff0c;下面演示一下入門示例 1. 配置Appium 1.1. 點擊Appium圖標&#xff0c;打開服務器&#xff1a; 1.2. 點擊Edit Configurations,進入配置頁面&am…

【LeetCode】【滑動窗口長度不固定】978 最長湍流子數組

1794.【軟件認證】最長的指定瑕疵度的元音子串 這個例題&#xff0c;是滑動窗口中長度不定求最大的題目&#xff0c;在看題之前可以先看一下【leetcode每日一題】【滑動窗口長度不固定】案例。 題目描述 定義&#xff1a;開頭和結尾都是元音字母&#xff08;aeiouAEIOU&…

Python腳本之打印乘法口訣表

Python腳本之打印乘法口訣表 for i in range(1, 10):for j in range(1, i 1):print(f"{j} * {i} {i * j}", end"\t")print()

GIT基本的命令

文章目錄 04.GIT本地操作-初始化工作區內容小結 05.GIT本地操作-add與commit目標內容小結 06.GIT本地操作-差異比較目標內容小結 07.GIT本地操作-版本回退目標內容小結 08.GIT本地操作-修改撤消目標內容小結 09.GIT本地操作-總結 04.GIT本地操作-初始化工作區 內容 初始化工作…

Java對象大小計算

概述 在實際應用中&#xff0c;尤其是在進行JVM調優時&#xff0c;理解并正確估計對象大小是非常重要的&#xff0c;因為這直接影響到內存分配、垃圾回收效率以及應用程序的整體性能。 對象的組成 在Java中&#xff0c;計算一個對象的大小是為了了解它在內存中占用的確切空間…

【c++基礎】挑戰賽第二題——放大的X

說明 請你編程畫一個放大的’X’&#xff08;大寫字母&#xff09;。 如3*3的’X’應如下所示&#xff1a; x xxx x 5*5的’X’如下所示&#xff1a; X XX XXX X X X 輸入數據 有一個正奇數n&#xff08;3 < n < 79&#xff09;&#xff0c;表示放大的規格。 …

m位數問題(c++題解)

題目描述 考官只給兩個整數n和m&#xff08;1 < n < 8&#xff0c;1< m <5&#xff09;&#xff0c;要求選手從1,2,…,n中取出m個數字&#xff0c;組成一個m位整數&#xff0c;統計所有的m位整數中一共有多少個素數。 如n3,m2時&#xff0c;符合條件的整數有&…

(C語言) time庫-日期和時間工具

文章目錄 ?介紹?常量??CLOCKS_PER_SEC ?類型??tm??time_t??clock_t??timespec (C11) ?函數-時間操作??time&#x1f3f7;?返回紀元開始經過的當前系統日歷時間??clock&#x1f3f7;?返回未加工的程序啟動時開始經過的處理器時間??difftime&#x1f3f7;?…

(delphi11最新學習資料) Object Pascal 學習筆記---第6章第3節(字符串連接)

6.3.3 字符串連接 ? 我已經提到過&#xff0c;與其他語言不同&#xff0c;Object Pascal 完全支持直接字符串連接&#xff0c;這實際上是一個相當快的操作。在本章中&#xff0c;我將向您展示一些字符串連接代碼和一些速度測試。稍后&#xff0c;在第 18 章中&#xff0c;我將…

第十五天-爬蟲項目實戰

目錄 1.介紹 2.代碼 1.main.py 2.PageSider.py 3.DetailSpider.py 4.DataParse.py 5.Constant.py 6.HanderRequest.py 1.介紹 1. 使用多線程爬取網站 2.爬取數據后保存至excel 3.爬取網站(僅做測試)網創類項目爬取&#xff1a;https://www.maomp.com/ 4..實現效果 …

python66-Python的循環之常用工具函數

使用zip()函數可以把兩個列表“壓縮”成一個zip對象(可迭代對象),這樣就可以使用一個循環并行遍歷兩個列表。為了測試 zip()函數的功能,我們可以先在交互式解釋器中“試驗”一下該函數的功能。 # !/usr/bin/env python# -*- coding: utf-8 -*-# @Time : 2024/01# @Author : …

元數據優化:提升您的網站在搜索引擎中的表現

前言 在之前的文章中&#xff0c;我們探討了如何通過超鏈接來提高用戶在網站的使用體驗。本篇將聚焦于元數據的優化&#xff0c;揭示它如何成為提升網站曝光率和點擊率的秘密武器。 一、介紹 元數據&#xff0c;或稱之為數據的數據&#xff0c;在網頁開發中占據著不可忽視的角…

IPD MM流程之業務策略工具:安索夫矩陣

IPD市場管理流程&#xff0c;華為內部稱為“MM流程”&#xff08;Market Management&#xff0c;MM&#xff09;。華為市場管理是通過對市場和細分市場的分析&#xff0c;制定細分市場的策略&#xff0c;形成商業計劃&#xff0c;把商業計劃落實在日常工作當中。MM流程其中一個…

git根據文件改動將文件自動添加到緩沖區

你需要修改以下腳本中的 use_cca: false 部分 #!/bin/bash# 獲取所有已修改但未暫存的文件 files$(git diff --name-only)for file in $files; do# 檢查文件中是否存在"use_cca: false"if grep -q "use_cca: false" "$file"; thenecho "Ad…

3.1線程作業

1.要求定義一個全局變量char buf"1234567"&#xff0c;創建兩個線程&#xff0c;不考慮退出條件。 a.A線程循環打印buf字符串&#xff0c; b.B線程循環倒置buf字符串&#xff0c;即buf中本來存儲1234567&#xff0c;倒置后buf中存儲7654321.B線程中不打印!! c.倒置…

qt5-入門-使用拖動方式創建Dialog

參考&#xff1a; C GUI Programming with Qt 4, Second Edition 本地環境&#xff1a; win10專業版&#xff0c;64位&#xff0c;Qt5.12 目錄 實現效果基本流程逐步實操1&#xff09;創建和初始化子部件2&#xff09;把子部件放進布局中3&#xff09;設置tab順序4&#xff09…

jstat命令查看jvm的GC信息

文章目錄 前言jstat命令查看jvm的GC信息1. 概述2. 應用堆內存水位閥值大小怎么確定3. 使用 jps 命令查看 Java 進程的進程號&#xff08;PID&#xff09;![在這里插入圖片描述](https://img-blog.csdnimg.cn/direct/5097401443314e9d808a83b694dbc6e5.png)4. jstat用法5. 類加載…

UE4 Niagara 關卡3.1官方案例解析二

自己嘗試做做&#xff0c;打亂順序 1、新建空的niagara system&#xff0c;添加空的發射器。更換渲染器為網格體渲染器并添加網格體。 2、發射器更新里面添加Spawn Rate&#xff0c;發射個粒子看看 效果圖&#xff1a; 3、采樣靜態網格體&#xff0c;網格體粒子出生于靜態網格…