mybatis開發一個分頁插件、mybatis實現分頁、mybatis攔截器

mybatis開發一個分頁插件、mybatis實現分頁、mybatis攔截器

通過官網的mybatis插件說明可知,我們可以通過攔截器進行開發一個插件。

例如這樣的:

        UserMapper mapper = sqlSession.getMapper(UserMapper.class);// 開始分頁MagicPage.startPage(1, 3);// 查詢List<UserEntity> list = mapper.selectAll();System.out.println(list);// 獲取到分頁的內容PageInfo page = MagicPage.getPage();System.out.println(page);

格式就是通過線程變量來實現

MagicPage.start(page,size);
// 執行查詢
...
MagicPage.getPage();

先編寫一個 MagicPage

import cn.hutool.core.lang.Assert;/*** @author lingkang* Created by 2024/3/3*/
public class MagicPage {private static final ThreadLocal<PageInfo> local = new ThreadLocal<>();/*** 開始分頁** @param page 默認 1* @param size 默認 10*/public static void startPage(int page, int size) {Assert.isTrue(page > 0, "page 最小值為 1");Assert.isTrue(size > 0, "size 必須大于 0");PageInfo info = new PageInfo();info.setSize(size);info.setPage(page);local.set(info);}public static PageInfo getPage() {return local.get();}public static void removePage() {local.remove();}
}

PageInfo

@Data
public class PageInfo {private int size;private int page;private long total;
}

mybatis插件實現:

import cn.hutool.core.convert.BasicType;
import org.apache.ibatis.executor.statement.StatementHandler;
import org.apache.ibatis.mapping.BoundSql;
import org.apache.ibatis.mapping.ParameterMapping;
import org.apache.ibatis.plugin.Interceptor;
import org.apache.ibatis.plugin.Intercepts;
import org.apache.ibatis.plugin.Invocation;
import org.apache.ibatis.plugin.Signature;import java.lang.reflect.Field;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.util.Collection;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;/*** 查詢分頁攔截** @author lingkang* Created by 2024/3/3*/
@Intercepts({@Signature(type = StatementHandler.class,method = "prepare",args = {Connection.class, Integer.class})})
public class MagicPageInterceptor implements Interceptor {@Overridepublic Object intercept(Invocation invocation) throws Throwable {PageInfo page = MagicPage.getPage();if (page == null)return invocation.proceed();StatementHandler statementHandler = (StatementHandler) invocation.getTarget();BoundSql boundSql = statementHandler.getBoundSql();if (!boundSql.getSql().toLowerCase().contains("select")) {return invocation.proceed();}Connection conn = (Connection) invocation.getArgs()[0];String totalSql = "select count(*) ";TotalConvert convert = pageSql(boundSql.getSql());if (convert.getOrderIndex() != 0)totalSql += boundSql.getSql().substring(convert.getTotalIndex()) + boundSql.getSql().substring(convert.getOrderIndex());elsetotalSql += boundSql.getSql().substring(convert.getTotalIndex());PreparedStatement statement = conn.prepareStatement(totalSql);Object parameterObject = boundSql.getParameterObject();// 設置參數if (parameterObject != null) {if (parameterObject.getClass().isPrimitive() || BasicType.WRAPPER_PRIMITIVE_MAP.containsKey(parameterObject.getClass())) {statement.setObject(1, parameterObject);} else if (parameterObject.getClass().isArray()) {Object[] arr = (Object[]) parameterObject;for (int i = 1; i <= arr.length; i++)statement.setObject(i, arr[i]);} else if (parameterObject instanceof Collection) {int i = 1;for (Object o : (Collection) parameterObject)statement.setObject(i++, o);} else if (parameterObject instanceof Map) {Map map = (Map) parameterObject;int i = 1;for (ParameterMapping mapping : boundSql.getParameterMappings())statement.setObject(i++, map.get(mapping.getProperty()));} else {// 將它當成對象int i = 1;for (ParameterMapping mapping : boundSql.getParameterMappings()) {Field field = parameterObject.getClass().getDeclaredField(mapping.getProperty());field.setAccessible(true);statement.setObject(i++, field.get(parameterObject));}}}ResultSet resultSet = statement.executeQuery();if (resultSet.next()) {long total = resultSet.getLong(1);page.setTotal(total);resultSet.close();}if (page.getTotal() > 0) {// 存在分頁String sql = boundSql.getSql() + " limit " + (page.getPage() - 1) * page.getSize() + "," + page.getSize();Field field = BoundSql.class.getDeclaredField("sql");field.setAccessible(true);field.set(boundSql, sql);}return invocation.proceed();}Pattern orderBy = Pattern.compile("order\\s+by");/*** select id,(select username from user where id=t_order.userid) as username from t_order order by id desc*/private TotalConvert pageSql(String sql) {TotalConvert convert = new TotalConvert();sql = sql.toLowerCase();int index = sql.indexOf("from");int start = sql.substring(0, index).indexOf("(");while (start != -1) {start = sql.indexOf("(", start + 1);if (start == -1)index = sql.indexOf("from", index + 1);}convert.setTotalIndex(index);// 匹配排序,可能存在排序,需要特殊處理Matcher matcher = orderBy.matcher(sql);if (matcher.find()) {convert.setOrderIndex(matcher.start());}return convert;}
}

需要注意,上面的插件只是做了mysql語法下的分頁,如果是sqlserver以及其他數據庫需要做調整

配置我們的插件

// 將攔截器添加到配置
configuration.addInterceptor(new MagicPageInterceptor());

執行效果:
在這里插入圖片描述

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

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

相關文章

Javascript:類型轉換

一、前言 prompt與表達單取過來的值默認為string類型 二、正文 1.隱式轉換 某些運算符被執行的時候&#xff0c;系統內部自動將數據類型進行轉換。 規則&#xff1a; 好兩邊只要有一個是字符串&#xff0c;都會把另外一個轉成字符串。 除了以外的算術運算符&#xff0c;比如…

Linux:線程的概念

個人主頁 &#xff1a; 個人主頁 個人專欄 &#xff1a; 《數據結構》 《C語言》《C》《Linux》 文章目錄 前言一、線程的概念線程代碼的簡單示例 總結 前言 本文是對于線程概念的知識總結 一、線程的概念 在課本上&#xff0c;線程是比進程更輕量級的一種指向流 或 線程是在…

VS Code 的粘性滾動預覽 - 類似于 Excel 的凍結首行

VS Code 的粘性滾動預覽 - 類似于 Excel 的凍結首行功能&#xff0c;即滾動 UI 顯示當前源代碼范圍。便于在代碼行數比較多的時候更好的知道自己所在的位置。粘性滾動UI 顯示用戶在滾動期間所處的范圍&#xff0c;將顯示編輯器頂部所在的類/接口/命名空間/函數/方法/構造函數&a…

4、Linux-常用命令(二)

目錄 一、搜索命令 1、命令搜索命令 2、文件搜索命令find。格式&#xff1a;find [搜索范圍] [搜索條件]。 3、字符串搜索命令grep 二、幫助命令 1、man【詳細的幫助】 2、--help【簡要的幫助】 三、壓縮與解壓命令 1、.zip格式 2、.gz格式 3、打包 四、關機和重啟命…

【大廠AI課學習筆記NO.57】(10)分類任務的評價指標

我們實際做的是一個分類任務。 在人工智能深度學習項目中&#xff0c;分類任務是指一種特定的任務類型&#xff0c;即預測結果是離散值的任務。具體來說&#xff0c;分類任務的目標是將輸入數據劃分到不同的類別中。這些類別可以是二分類&#xff08;如垃圾郵件分類&#xff0c…

理解這幾個安全漏洞,你也能做安全測試

01 短信炸彈 1、漏洞描述 短信轟炸攻擊是常見的一種攻擊&#xff0c;攻擊者通過網站頁面中所提供的發送短信驗證碼的功能處&#xff0c;通過對其發送數據包的獲取后&#xff0c;進行重放&#xff0c;如果服務器短信平臺未做校驗的情況時&#xff0c;系統會一直去發送短信&…

函數式響應式編程(FRP):構筑靈活動態的應用程序

FRP&#xff08;Functional Reactive Programming&#xff0c;函數式響應式編程&#xff09;是一個編程范式&#xff0c;它結合了函數式編程和響應式編程的原則&#xff0c;用于處理時間變化的數據和響應性系統。FRP 讓開發者能夠以聲明式地方式來構建響應用戶輸入、網絡請求或…

【vue3 路由使用與講解】vue-router : 簡潔直觀的全面介紹

# 核心內容介紹 路由跳轉有兩種方式&#xff1a; 聲明式導航&#xff1a;<router-link :to"...">編程式導航&#xff1a;router.push(...) 或 router.replace(...) &#xff1b;兩者的規則完全一致。 push(to: RouteLocationRaw): Promise<NavigationFailur…

JVM內部世界(內存劃分,類加載,垃圾回收)

&#x1f495;"Echo"&#x1f495; 作者&#xff1a;Mylvzi 文章主要內容&#xff1a;JVM內部世界(內存劃分,類加載,垃圾回收) 關于JVM的學習主要掌握三方面: JVM內存區的劃分類加載垃圾回收 一.JVM內存區的劃分 當一個Java進程開始執行時,JVM會首先向操作系統申…

實例驅動計算機網絡

文章目錄 計算機網絡的層次結構應用層DNSHTTP協議HTTP請求響應過程 運輸層TCP協議TCP協議面向連接實現TCP的三次握手連接TCP的四次揮手斷開連接 TCP協議可靠性實現TCP的流量控制TCP的擁塞控制TCP的重傳機制 UDP協議 網際層IP協議&#xff08;主機與主機&#xff09;IP地址的分類…

php 讀取文件并以文件方式下載

if (!file_exists($filename)){//判斷能否獲取這個文件header("Content-type: text/html; charset=utf-8");echo "File not found!";exit

【創作回顧】17個月崢嶸創作史

#里程碑專區#、#創作者紀念日# 還記得 2022 年 10 月 05 日&#xff0c;我在CSDN撰寫了第 1 篇博客——《關于測試工程師瓶頸和突圍的一個思考》&#xff0c;也是我在全網發布的第一篇技術文章。 回想當時&#xff0c;這一篇的誕生過程并不輕松&#xff0c;不像是一篇網絡文章…

【計算機網絡】深度學習HTTPS協議

&#x1f493; 博客主頁&#xff1a;從零開始的-CodeNinja之路 ? 收錄文章&#xff1a;【計算機網絡】深度學習HTTPS協議 &#x1f389;歡迎大家點贊&#x1f44d;評論&#x1f4dd;收藏?文章 目錄 一:HTTPS是什么二:HTTPS的工作過程三:對稱加密四:非對稱加密五:中間人攻擊1…

【web | CTF】BUUCTF [HCTF 2018]WarmUp

天命&#xff1a;這題本地php代碼是無法復現的 首先打開網站&#xff0c;啥也沒有&#xff0c;查看源碼 發現文件&#xff0c;打開訪問一下看看&#xff0c;發現是代碼審計 <?phphighlight_file(__FILE__);class emmm{public static function checkFile(&$page){$whit…

【學習總結】什么是DoS和DDoS

[Q&A] 什么是DoS DoS 是 “Denial of Service”&#xff08;拒絕服務&#xff09;的縮寫&#xff0c;它是一種網絡攻擊方式&#xff0c;其目的是使目標計算機或網絡資源無法為合法用戶提供正常的服務。通過向目標系統發送大量請求、消耗其帶寬、處理器或內存等資源&#…

13 雙口 RAM IP 核

雙口 RAM IP 核簡介 雙口 RAM IP 核有兩個端口&#xff0c;它又分為偽雙端口 RAM 和真雙端口 RAM&#xff0c;偽雙端口 RAM 一個端口只能讀&#xff0c;另一個端口只能 寫&#xff0c;真雙端口 RAM 兩個端口都可以進行讀寫操作。同時對存儲器進行讀寫操作時就會用到雙端口 RAM…

unity-1

創建游戲對象&#xff08;游戲物體&#xff09; 可通過unity中的菜單欄中的Gameobject創建&#xff1b;也可在Hierarchy&#xff08;層級&#xff09;中創建&#xff0c; 雙擊即可居中看到。 在Hierarchy空白處右鍵即可看到&#xff0c;能創建游戲對象。 在Scene框中&#x…

BioTech - ADMET的性質預測 概述

歡迎關注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://blog.csdn.net/caroline_wendy/article/details/136438192 ADMET&#xff0c;即 Absorption、Distribution、Metabolism、Excretion、Toxicity&#xff0c;吸收、分布、代謝、排泄、毒性…

題目 1629: 藍橋杯算法訓練VIP-接水問題

題目描述: 學校里有一個水房&#xff0c;水房里一共裝有m個龍頭可供同學們打開水&#xff0c;每個龍頭每秒鐘的供水量相等&#xff0c;均為1。現在有n名同學準備接水&#xff0c;他們的初始接水順序已經確定。將這些同學按接水順序從1到n編號&#xff0c;i號同學的接水量為wi。…

Linux shell:補充命令的使用

目錄 一.導讀 二.正文 三.結語 一.導讀 上一篇介紹了腳本的簡單概念以及使用&#xff0c;現在補充一些命令。 二.正文 目前處于全局目錄&#xff0c;通過mkdir創建名我為day01的文件。 通過cd命令day01 切換至day01文件當中。 使用vim文本編輯器文件名&#xff08;firstdir&…