Mybatis-Plus實(shí)現(xiàn)SQL攔截器的示例
起源
最近公司要做多租戶(hù),Mybatis-Plus的多租戶(hù)插件很好用,但是有一個(gè)場(chǎng)景是:字典表或者某些數(shù)據(jù)表,一些數(shù)據(jù)需要在各個(gè)租戶(hù)之間共享,而數(shù)據(jù)表本身又同時(shí)要實(shí)現(xiàn)多租戶(hù)數(shù)據(jù)隔離,比如字典表:性別等“男”/“女”基礎(chǔ)數(shù)據(jù)。
SQL攔截器是一種用于攔截和修改Mybatis執(zhí)行的SQL語(yǔ)句的工具。通過(guò)使用SQL攔截器,開(kāi)發(fā)人員可以在執(zhí)行SQL語(yǔ)句之前或之后對(duì)其進(jìn)行修改或記錄,從而更好地控制和優(yōu)化數(shù)據(jù)庫(kù)操作。例如MyBatis-Plus的基礎(chǔ)分頁(yè)插件、多租戶(hù)插件就是SQL攔截器,那么,我們是否可以實(shí)現(xiàn)自己的SQL攔截器呢?
答案當(dāng)然是肯定的。
實(shí)現(xiàn)攔截器接口InnerInterceptor
InnerInterceptor 這個(gè)接口是MyBaitsPlus的攔截器接口類(lèi),實(shí)現(xiàn)它之后,并且通過(guò)MybatisPlusInterceptor配置后,就可以實(shí)現(xiàn)SQL執(zhí)行攔截。
MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor(); // 多租戶(hù)攔截器 interceptor.addInnerInterceptor(new TenantLineInnerInterceptor(new TenantDatabaseInterceptor(tenantProperties))); // 多租戶(hù)部分表數(shù)據(jù)共享攔截器 interceptor.addInnerInterceptor(new TenantDatabaseShareInterceptor(new TenantShareHandlerImpl(tenantProperties)));
那么如何實(shí)現(xiàn)該接口以及修改SQL呢?
下面是一個(gè)樣例:
import com.baomidou.mybatisplus.core.interceptor.InnerInterceptor;
import com.baomidou.mybatisplus.core.metadata.IPage;
import com.baomidou.mybatisplus.core.parser.SqlParserHelper;
import com.baomidou.mybatisplus.extension.plugins.inner.InnerInterceptorChain;
import net.sf.jsqlparser.parser.CCJSqlParserUtil;
import net.sf.jsqlparser.statement.Statement;
import net.sf.jsqlparser.statement.select.Select;
public class MyInterceptor implements InnerInterceptor {
@Override
```
@Override
public void beforeQuery(Executor executor, MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) throws SQLException {
if (InterceptorIgnoreHelper.willIgnoreTenantLine(ms.getId())) return;
PluginUtils.MPBoundSql mpBs = PluginUtils.mpBoundSql(boundSql);
mpBs.sql(parserSingle(mpBs.sql(), null));
}
}通過(guò)修改beforeQuery函數(shù)中的代碼,即可修改執(zhí)行的SQL。
修改sql常用的工具類(lèi)
JsqlParserSupport該類(lèi)用于解析與修改SQL,并且MybatisPlus的多租戶(hù)插件更是直接繼承了該類(lèi)。 CCJSqlParserUtil是JSqlParser中用于解析SQL語(yǔ)句的工具類(lèi)。它提供了一些靜態(tài)方法,可以將SQL語(yǔ)句解析為Statement對(duì)象、Select對(duì)象、Update對(duì)象、Insert對(duì)象、Delete對(duì)象等。
常用的類(lèi)還有Expression, Statement等類(lèi)。
例如,使用Expression實(shí)現(xiàn)一個(gè)IN語(yǔ)句:
import net.sf.jsqlparser.expression.Expression;
import net.sf.jsqlparser.expression.LongValue;
import net.sf.jsqlparser.expression.StringValue;
import net.sf.jsqlparser.expression.operators.relational.InExpression;
import net.sf.jsqlparser.expression.operators.relational.ItemsList;
import net.sf.jsqlparser.expression.operators.relational.MultiExpressionList;
import net.sf.jsqlparser.expression.operators.relational.NamedParameter;
import net.sf.jsqlparser.expression.operators.relational.SubSelect;
import net.sf.jsqlparser.parser.CCJSqlParserUtil;
public class JSqlParserExample {
public static void main(String[] args) throws Exception {
// IN語(yǔ)句
InExpression inExpr = new InExpression();
inExpr.setLeftExpression(CCJSqlParserUtil.parseExpression("age"));
ItemsList valueList = new MultiExpressionList();
((MultiExpressionList) valueList).addExpressions(new LongValue(18), new LongValue(21), new LongValue(30));
inExpr.setItemsList(valueList);
System.out.println(inExpr.toString());
// 子查詢(xún)
InExpression subqueryInExpr = new InExpression();
subqueryInExpr.setLeftExpression(CCJSqlParserUtil.parseExpression("age"));
SubSelect subquery = new SubSelect();
subquery.setSelectBody(CCJSqlParserUtil.parse("SELECT age FROM users WHERE country = 'US'"));
subqueryInExpr.setRightExpression(subquery);
System.out.println(subqueryInExpr.toString());
}
}到此這篇關(guān)于Mybatis-Plus實(shí)現(xiàn)一個(gè)SQL攔截器的文章就介紹到這了,更多相關(guān)Mybatis-Plus SQL攔截器內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- MybatisPlusInterceptor實(shí)現(xiàn)sql攔截器超詳細(xì)教程
- Mybatis之如何攔截慢SQL日志記錄
- 利用Mybatis?Plus實(shí)現(xiàn)一個(gè)SQL攔截器
- Mybatis攔截器打印sql問(wèn)題
- MyBatis如何通過(guò)攔截修改SQL
- spring?boot?Mybatis?攔截器實(shí)現(xiàn)拼接sql和修改的代碼詳解
- MyBatis自定義SQL攔截器示例詳解
- mybatis 通過(guò)攔截器打印完整的sql語(yǔ)句以及執(zhí)行結(jié)果操作
- mybatis 實(shí)現(xiàn) SQL 查詢(xún)攔截修改詳解
- MyBatis的動(dòng)態(tài)攔截sql并修改
相關(guān)文章
Java連接redis報(bào)錯(cuò)timed?out問(wèn)題的解決辦法
最近項(xiàng)目開(kāi)發(fā)中用到了Redis,下面這篇文章主要給大家介紹了關(guān)于Java連接redis報(bào)錯(cuò)timed?out問(wèn)題的解決辦法,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-02-02
Java中Spock框架Mock對(duì)象的方法經(jīng)驗(yàn)總結(jié)
這篇文章主要分享了Spock框架Mock對(duì)象的方法經(jīng)驗(yàn)總結(jié),下文分享一些常用項(xiàng)目實(shí)戰(zhàn)說(shuō)明以及代碼,供大家項(xiàng)目中參考,也具有一的的參考價(jià)值,需要的小伙伴可以參考一下2022-02-02
Java中的Semaphore信號(hào)量使用方法代碼實(shí)例
這篇文章主要介紹了Java中的Semaphore信號(hào)量使用方法代碼實(shí)例,Semaphore是一種基于計(jì)數(shù)的信號(hào)量,它可以設(shè)定一個(gè)閾值,基于此,多個(gè)線(xiàn)程競(jìng)爭(zhēng)獲取許可信號(hào),做自己的申請(qǐng)后歸還,超過(guò)閾值后,線(xiàn)程申請(qǐng)?jiān)S可信號(hào)將會(huì)被阻塞,需要的朋友可以參考下2023-11-11
java四種引用及在LeakCanery中應(yīng)用詳解
這篇文章主要介紹了java四種引用及在LeakCanery中應(yīng)用,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-09-09
Java實(shí)現(xiàn)批量修改txt文件名稱(chēng)的方法示例
這篇文章主要介紹了Java實(shí)現(xiàn)批量修改txt文件名稱(chēng)的方法,結(jié)合實(shí)例形式分析了Java針對(duì)目錄文件遍歷及文件讀寫(xiě)、屬性操作等相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2019-03-03
SpringBoot項(xiàng)目使用yml文件鏈接數(shù)據(jù)庫(kù)異常問(wèn)題解決方案
在使用SpringBoot時(shí),利用yml進(jìn)行數(shù)據(jù)庫(kù)連接配置需小心數(shù)據(jù)類(lèi)型區(qū)分,如果用戶(hù)名或密碼是數(shù)字,必須用雙引號(hào)包裹以識(shí)別為字符串,避免連接錯(cuò)誤,特殊字符密碼也應(yīng)用引號(hào)包裹2024-10-10

