0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

基于Mybatis攔截器實(shí)現(xiàn)數(shù)據(jù)范圍權(quán)限

jf_ro2CN3Fa ? 來源:稀土掘金技術(shù)社區(qū) ? 2023-06-20 09:57 ? 次閱讀

mybatis 攔截器

mybatis攔截器優(yōu)先級(jí)

@Order

@DependsOn

@PostConstruct

ApplicationRunner

前端的菜單和按鈕權(quán)限都可以通過配置來實(shí)現(xiàn),但很多時(shí)候,后臺(tái)查詢數(shù)據(jù)庫(kù)數(shù)據(jù)的權(quán)限需要通過手動(dòng)添加SQL來實(shí)現(xiàn)。

比如員工打卡記錄表,有 id、name、dpt_id、company_id 等字段,后兩個(gè)表示部門 ID 和分公司 ID。

查看員工打卡記錄 SQL 為:select id,name,dpt_id,company_id from t_record

當(dāng)一個(gè)總部賬號(hào)可以查看全部數(shù)據(jù)此時(shí),sql 無需改變。因?yàn)樗梢钥吹饺繑?shù)據(jù)。

當(dāng)一個(gè)部門管理員權(quán)限員工查看全部數(shù)據(jù)時(shí),sql 需要在末屬添加 where dpt_id = #{dpt_id}

如果每個(gè)功能模塊都需要手動(dòng)寫代碼去拿到當(dāng)前登陸用戶的所屬部門,然后手動(dòng)添加where條件,就顯得非常的繁瑣。

因此,可以通過 mybatis 的攔截器拿到查詢 sql 語句,再自動(dòng)改寫 sql。

mybatis 攔截器

MyBatis 允許你在映射語句執(zhí)行過程中的某一點(diǎn)進(jìn)行攔截調(diào)用。默認(rèn)情況下,MyBatis 允許使用插件來攔截的方法調(diào)用包括:

Executor (update, query, flushStatements, commit, rollback, getTransaction, close, isClosed)

ParameterHandler (getParameterObject, setParameters)

ResultSetHandler (handleResultSets, handleOutputParameters)

StatementHandler (prepare, parameterize, batch, update, query)

這些類中方法的細(xì)節(jié)可以通過查看每個(gè)方法的簽名來發(fā)現(xiàn),或者直接查看 MyBatis 發(fā)行包中的源代碼。如果你想做的不僅僅是監(jiān)控方法的調(diào)用,那么你最好相當(dāng)了解要重寫的方法的行為。因?yàn)樵谠噲D修改或重寫已有方法的行為時(shí),很可能會(huì)破壞 MyBatis 的核心模塊。這些都是更底層的類和方法,所以使用插件的時(shí)候要特別當(dāng)心。

通過 MyBatis 提供的強(qiáng)大機(jī)制,使用插件是非常簡(jiǎn)單的,只需實(shí)現(xiàn) Interceptor 接口,并指定想要攔截的方法簽名即可。

分頁插件 pagehelper 就是一個(gè)典型的通過攔截器去改寫 SQL 的。

855e0d7e-0f0a-11ee-962d-dac502259ad0.jpg

可以看到它通過注解 @Intercepts 和簽名 @Signature 來實(shí)現(xiàn),攔截 Executor 執(zhí)行器,攔截所有的 query 查詢類方法。

我們可以據(jù)此也實(shí)現(xiàn)自己的攔截器。

importcom.skycomm.common.util.user.Cpip2UserDeptVo;
importcom.skycomm.common.util.user.Cpip2UserDeptVoUtil;
importlombok.extern.slf4j.Slf4j;
importorg.apache.commons.lang3.StringUtils;
importorg.apache.ibatis.cache.CacheKey;
importorg.apache.ibatis.executor.Executor;
importorg.apache.ibatis.mapping.BoundSql;
importorg.apache.ibatis.mapping.MappedStatement;
importorg.apache.ibatis.mapping.SqlSource;
importorg.apache.ibatis.plugin.Interceptor;
importorg.apache.ibatis.plugin.Intercepts;
importorg.apache.ibatis.plugin.Invocation;
importorg.apache.ibatis.plugin.Signature;
importorg.apache.ibatis.session.ResultHandler;
importorg.apache.ibatis.session.RowBounds;
importorg.springframework.stereotype.Component;
importorg.springframework.web.context.request.RequestAttributes;
importorg.springframework.web.context.request.RequestContextHolder;
importorg.springframework.web.context.request.ServletRequestAttributes;

importjavax.servlet.http.HttpServletRequest;
importjava.lang.reflect.Method;

@Component
@Intercepts({
@Signature(type=Executor.class,method="query",args={MappedStatement.class,Object.class,RowBounds.class,ResultHandler.class}),
@Signature(type=Executor.class,method="query",args={MappedStatement.class,Object.class,RowBounds.class,ResultHandler.class,CacheKey.class,BoundSql.class}),
})
@Slf4j
publicclassMySqlInterceptorimplementsInterceptor{

@Override
publicObjectintercept(Invocationinvocation)throwsThrowable{
MappedStatementstatement=(MappedStatement)invocation.getArgs()[0];
Objectparameter=invocation.getArgs()[1];
BoundSqlboundSql=statement.getBoundSql(parameter);
StringoriginalSql=boundSql.getSql();
ObjectparameterObject=boundSql.getParameterObject();

SqlLimitsqlLimit=isLimit(statement);
if(sqlLimit==null){
returninvocation.proceed();
}

RequestAttributesreq=RequestContextHolder.getRequestAttributes();
if(req==null){
returninvocation.proceed();
}

//處理request
HttpServletRequestrequest=((ServletRequestAttributes)req).getRequest();
Cpip2UserDeptVouserVo=Cpip2UserDeptVoUtil.getUserDeptInfo(request);
StringdepId=userVo.getDeptId();

Stringsql=addTenantCondition(originalSql,depId,sqlLimit.alis());
log.info("原SQL:{},數(shù)據(jù)權(quán)限替換后的SQL:{}",originalSql,sql);
BoundSqlnewBoundSql=newBoundSql(statement.getConfiguration(),sql,boundSql.getParameterMappings(),parameterObject);
MappedStatementnewStatement=copyFromMappedStatement(statement,newBoundSqlSqlSource(newBoundSql));
invocation.getArgs()[0]=newStatement;
returninvocation.proceed();
}

/**
*重新拼接SQL
*/
privateStringaddTenantCondition(StringoriginalSql,StringdepId,Stringalias){
Stringfield="dpt_id";
if(StringUtils.isNoneBlank(alias)){
field=alias+"."+field;
}

StringBuildersb=newStringBuilder(originalSql);
intindex=sb.indexOf("where");
if(indexcls=Class.forName(className);
finalMethod[]method=cls.getMethods();
for(Methodme:method){
if(me.getName().equals(methodName)&&me.isAnnotationPresent(SqlLimit.class)){
sqlLimit=me.getAnnotation(SqlLimit.class);
}
}
}catch(Exceptione){
e.printStackTrace();
}
returnsqlLimit;
}

publicstaticclassBoundSqlSqlSourceimplementsSqlSource{

privatefinalBoundSqlboundSql;

publicBoundSqlSqlSource(BoundSqlboundSql){
this.boundSql=boundSql;
}

@Override
publicBoundSqlgetBoundSql(ObjectparameterObject){
returnboundSql;
}
}
}

順便加了個(gè)注解 @SqlLimit,在 mapper 方法上加了此注解才進(jìn)行數(shù)據(jù)權(quán)限過濾。同時(shí)注解有兩個(gè)屬性,

@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
public@interfaceSqlLimit{
/**
*sql表別名
*@return
*/
Stringalis()default"";

/**
*通過此列名進(jìn)行限制
*@return
*/
StringcolumnName()default"";
}

columnName 表示通過此列名進(jìn)行限制,一般來說一個(gè)系統(tǒng),各表當(dāng)中的此列是統(tǒng)一的,可以忽略。

alis 用于標(biāo)注 sql 表別名,如 針對(duì) sql select * from tablea as a left join tableb as b on a.id = b.id 進(jìn)行改寫,如果不知道表別名,會(huì)直接在后面拼接 where dpt_id = #{dptId},那此 SQL 就會(huì)錯(cuò)誤的,通過別名 @SqlLimit(alis = "a") 就可以知道需要拼接的是 where a.dpt_id = #{dptId}

執(zhí)行結(jié)果。

原 SQL:select * from person, 數(shù)據(jù)權(quán)限替換后的SQL:select * from person where dpt_id = 234。

原 SQL:select * from person where id > 1, 數(shù)據(jù)權(quán)限替換后的 SQL:select * from person where dpt_id = 234 and id > 1。

但是在使用 PageHelper 進(jìn)行分頁的時(shí)候還是有問題。

857effca-0f0a-11ee-962d-dac502259ad0.jpg

可以看到先執(zhí)行了 _COUNT 方法也就是 PageHelper,再執(zhí)行了自定義的攔截器。

在我們的業(yè)務(wù)方法中注入 SqlSessionFactory。

@Autowired
@Lazy
privateListsqlSessionFactoryList;
859abcba-0f0a-11ee-962d-dac502259ad0.jpg

PageInterceptor 為 1,自定義攔截器為 0,跟 order 相反,PageInterceptor 優(yōu)先級(jí)更高,所以越先執(zhí)行。

基于 Spring Boot + MyBatis Plus + Vue & Element 實(shí)現(xiàn)的后臺(tái)管理系統(tǒng) + 用戶小程序,支持 RBAC 動(dòng)態(tài)權(quán)限、多租戶、數(shù)據(jù)權(quán)限、工作流、三方登錄、支付、短信、商城等功能

項(xiàng)目地址:https://github.com/YunaiV/ruoyi-vue-pro

視頻教程:https://doc.iocoder.cn/video/

mybatis攔截器優(yōu)先級(jí)

@Order

通過 @Order 控制 PageInterceptor 和 MySqlInterceptor 可行嗎?

85ad7ca6-0f0a-11ee-962d-dac502259ad0.jpg

將 MySqlInterceptor 的加載優(yōu)先級(jí)調(diào)到最高,但測(cè)試證明依然不行。

定義 3 個(gè)類。

@Component
@Order(2)
publicclassOrderTest1{

@PostConstruct
publicvoidinit(){
System.out.println("00000init");
}
}
@Component
@Order(1)
publicclassOrderTest2{

@PostConstruct
publicvoidinit(){
System.out.println("00001init");
}
}
@Component
@Order(0)
publicclassOrderTest3{

@PostConstruct
publicvoidinit(){
System.out.println("00002init");
}
}

OrderTest1,OrderTest2,OrderTest3 的優(yōu)先級(jí)從低到高。

順序預(yù)期的執(zhí)行順序應(yīng)該是相反的:

00002init
00001init
00000init

但事實(shí)上執(zhí)行的順序是

00000init
00001init
00002init

@Order 不控制實(shí)例化順序,只控制執(zhí)行順序。@Order 只跟特定一些注解生效 如:@Compent、 @Service、@Aspect … 不生效的如:@WebFilter

所以這里達(dá)不到預(yù)期效果。

@Priority 類似,同樣不行。

@DependsOn

使用此注解將當(dāng)前類將在依賴類實(shí)例化之后再執(zhí)行實(shí)例化。

在 MySqlInterceptor 上標(biāo)記@DependsOn("queryInterceptor")

85cec4a6-0f0a-11ee-962d-dac502259ad0.jpg

啟動(dòng)報(bào)錯(cuò),

這個(gè)時(shí)候 queryInterceptor 還沒有實(shí)例化對(duì)象。

@PostConstruct

@PostConstruct 修飾的方法會(huì)在服務(wù)器加載 Servlet 的時(shí)候運(yùn)行,并且只會(huì)被服務(wù)器執(zhí)行一次。在同一個(gè)類里,執(zhí)行順序?yàn)轫樞蛉缦拢篊onstructor > @Autowired > @PostConstruct。

但它也不能保證不同類的執(zhí)行順序。

PageHelper 的 springboot start 也是通過這個(gè)來初始化攔截器的。

85e3f43e-0f0a-11ee-962d-dac502259ad0.jpg

ApplicationRunner

在當(dāng)前 springboot 容器加載完成后執(zhí)行,那么這個(gè)時(shí)候 pagehelper 的攔截器已經(jīng)加入,在這個(gè)時(shí)候加入自定義攔截器,就能達(dá)到我們想要的效果。

仿照 PageHelper 來寫。

@Component
publicclassInterceptRunnerimplementsApplicationRunner{

@Autowired
privateListsqlSessionFactoryList;

@Override
publicvoidrun(ApplicationArgumentsargs)throwsException{
MySqlInterceptormybatisInterceptor=newMySqlInterceptor();
for(SqlSessionFactorysqlSessionFactory:sqlSessionFactoryList){
org.apache.ibatis.session.Configurationconfiguration=sqlSessionFactory.getConfiguration();
configuration.addInterceptor(mybatisInterceptor);
}
}
}

再執(zhí)行,可以看到自定義攔截器在攔截器鏈當(dāng)中下標(biāo)變?yōu)榱?1(優(yōu)先級(jí)與 order 剛好相反)

860ace9c-0f0a-11ee-962d-dac502259ad0.jpg

后臺(tái)打印結(jié)果,達(dá)到了預(yù)期效果。

8631d384-0f0a-11ee-962d-dac502259ad0.jpg

審核編輯:湯梓紅
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 接口
    +關(guān)注

    關(guān)注

    33

    文章

    8611

    瀏覽量

    151247
  • 數(shù)據(jù)
    +關(guān)注

    關(guān)注

    8

    文章

    7048

    瀏覽量

    89076
  • SQL
    SQL
    +關(guān)注

    關(guān)注

    1

    文章

    764

    瀏覽量

    44156
  • mybatis
    +關(guān)注

    關(guān)注

    0

    文章

    60

    瀏覽量

    6716
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    Mybatis 攔截器實(shí)現(xiàn)數(shù)據(jù)源內(nèi)多數(shù)據(jù)庫(kù)切換

    數(shù)據(jù)庫(kù) 現(xiàn)在需要上線報(bào)表服務(wù)來查詢所有數(shù)據(jù)庫(kù)中的數(shù)據(jù)進(jìn)行統(tǒng)計(jì),那么現(xiàn)在的問題來了,該如何 滿足在配置一個(gè)數(shù)據(jù)源的情況下來查詢?cè)?b class='flag-5'>數(shù)據(jù)源下不同
    的頭像 發(fā)表于 12-12 10:23 ?735次閱讀

    HarmonyOS實(shí)戰(zhàn)開發(fā)-如何在Navigation中完成路由攔截

    路由攔截器interceptor.ets,定義攔截容器、注冊(cè)方法和公共攔截邏輯,interceptor.ets /** * 定義攔截實(shí)現(xiàn)
    發(fā)表于 05-08 14:21

    [推薦]奧運(yùn)全球眼 網(wǎng)絡(luò)視頻攝像機(jī) 電話報(bào)警系統(tǒng) -電話報(bào)警 GSM防盜

    ................................................................. 414.3.4 會(huì)話bean 的生命周期回調(diào)攔截器方法 ................................. 424.3.5
    發(fā)表于 07-07 15:39

    我想做一個(gè)號(hào)碼攔截器。面對(duì)面5米內(nèi)接收到對(duì)方的手機(jī)號(hào)碼。我也咨詢很多人,不是技

    我想做一個(gè)號(hào)碼攔截器。面對(duì)面5米內(nèi)接收到對(duì)方的手機(jī)號(hào)碼。我也咨詢很多人,不是技術(shù)問題就是,怕這東西觸犯法律。我只是正規(guī)用途,并不會(huì)觸犯法律底線!望“能人”解決我的問題!樣品只要符合以上條件,重金酬謝...謝謝!QQ896776242加我請(qǐng)注明電子*** 丁先生
    發(fā)表于 04-29 16:16

    網(wǎng)絡(luò)組件axios可以在OpenHarmony上使用了

    攔截器也是如此功能,只是在請(qǐng)求得到響應(yīng)之后,對(duì)響應(yīng)體的一些處理,通常是數(shù)據(jù)統(tǒng)一處理等,也常來判斷登錄失效等。axios的攔截器作用非常大。axios的攔截器分為請(qǐng)求
    發(fā)表于 08-29 12:11

    動(dòng)能攔截器六自由度仿真建模研究

    仿真建模技術(shù)是動(dòng)能攔截器制導(dǎo)律研究中的重要技術(shù),文中主要建立動(dòng)能攔截器的軌道運(yùn)動(dòng)動(dòng)力學(xué)以及姿態(tài)運(yùn)動(dòng)動(dòng)力學(xué)模型,并建立完整的制導(dǎo)控制系統(tǒng)數(shù)學(xué)模型。文末,以某型
    發(fā)表于 08-07 08:50 ?14次下載

    利用API攔截技術(shù)實(shí)現(xiàn)串口通信數(shù)據(jù)攔截

    文中介紹了API 攔截技術(shù)的基本原理和應(yīng)用框架,研究了實(shí)現(xiàn)攔截Windows API 函數(shù)的一種方法,最后詳細(xì)說明了利用此法實(shí)現(xiàn)串口通信數(shù)據(jù)
    發(fā)表于 05-13 15:51 ?24次下載
    利用API<b class='flag-5'>攔截</b>技術(shù)<b class='flag-5'>實(shí)現(xiàn)</b>串口通信<b class='flag-5'>數(shù)據(jù)</b><b class='flag-5'>攔截</b>

    springmvc 自定義攔截器實(shí)現(xiàn)未登錄用戶的攔截

    springmvc自定義攔截器實(shí)現(xiàn)未登錄用戶的攔截
    發(fā)表于 11-25 14:44 ?2520次閱讀
    springmvc 自定義<b class='flag-5'>攔截器</b><b class='flag-5'>實(shí)現(xiàn)</b>未登錄用戶的<b class='flag-5'>攔截</b>

    MyBatis實(shí)現(xiàn)原理

    本文主要詳細(xì)介紹了MyBatis實(shí)現(xiàn)原理。mybatis底層還是采用原生jdbc來對(duì)數(shù)據(jù)庫(kù)進(jìn)行操作的,只是通過 SqlSessionFactory,SqlSession Execut
    的頭像 發(fā)表于 02-24 11:25 ?6493次閱讀
    <b class='flag-5'>MyBatis</b>的<b class='flag-5'>實(shí)現(xiàn)</b>原理

    SpringBoot+Mybatis如何實(shí)現(xiàn)流式查詢?

    使用mybatis作為持久層的框架時(shí),通過mybatis執(zhí)行查詢數(shù)據(jù)的請(qǐng)求執(zhí)行成功后,mybatis返回的結(jié)果集不是一個(gè)集合或?qū)ο?,而是一個(gè)迭代
    的頭像 發(fā)表于 06-12 09:57 ?1281次閱讀

    如何實(shí)現(xiàn)基于Mybatis攔截器實(shí)現(xiàn)數(shù)據(jù)范圍權(quán)限呢?

    前端的菜單和按鈕權(quán)限都可以通過配置來實(shí)現(xiàn),但很多時(shí)候,后臺(tái)查詢數(shù)據(jù)庫(kù)數(shù)據(jù)權(quán)限需要通過手動(dòng)添加SQL來實(shí)
    的頭像 發(fā)表于 06-20 09:59 ?1246次閱讀
    如何<b class='flag-5'>實(shí)現(xiàn)</b>基于<b class='flag-5'>Mybatis</b><b class='flag-5'>攔截器</b><b class='flag-5'>實(shí)現(xiàn)</b><b class='flag-5'>數(shù)據(jù)</b><b class='flag-5'>范圍</b><b class='flag-5'>權(quán)限</b>呢?

    如何利用MyBatis Plus去實(shí)現(xiàn)數(shù)據(jù)權(quán)限控制呢?

    平時(shí)開發(fā)中遇到根據(jù)當(dāng)前用戶的角色,只能查看數(shù)據(jù)權(quán)限范圍數(shù)據(jù)需求。列表實(shí)現(xiàn)方案有兩種,一是在開發(fā)初期就做好判斷賽選,但如果這個(gè)需求是中途加的
    的頭像 發(fā)表于 08-23 10:40 ?1067次閱讀
    如何利用<b class='flag-5'>MyBatis</b> Plus去<b class='flag-5'>實(shí)現(xiàn)</b><b class='flag-5'>數(shù)據(jù)</b><b class='flag-5'>權(quán)限</b>控制呢?

    springboot過濾器和攔截器哪個(gè)先執(zhí)行

    Spring Boot是一個(gè)用于構(gòu)建Java應(yīng)用程序的開發(fā)框架,它提供了許多功能和工具來簡(jiǎn)化開發(fā)和部署過程。其中兩個(gè)重要的功能是過濾器和攔截器。本文將詳細(xì)介紹Spring Boot過濾器和攔截器
    的頭像 發(fā)表于 12-03 15:00 ?2558次閱讀

    使用go語言實(shí)現(xiàn)一個(gè)grpc攔截器

    在開發(fā)grpc服務(wù)時(shí),我們經(jīng)常會(huì)遇到一些通用的需求,比如:日志、鏈路追蹤、鑒權(quán)等。這些需求可以通過grpc攔截器實(shí)現(xiàn)。本文使用go語言來實(shí)現(xiàn)一個(gè) grpc一元模式(Unary)攔截器
    的頭像 發(fā)表于 12-18 10:13 ?698次閱讀
    使用go語言<b class='flag-5'>實(shí)現(xiàn)</b>一個(gè)grpc<b class='flag-5'>攔截器</b>

    使用mybatis切片實(shí)現(xiàn)數(shù)據(jù)權(quán)限控制

    一、使用方式 數(shù)據(jù)權(quán)限控制需要對(duì)查詢出的數(shù)據(jù)進(jìn)行篩選,對(duì)業(yè)務(wù)入侵最少的方式就是利用mybatis或者數(shù)據(jù)庫(kù)連接池的切片對(duì)已有業(yè)務(wù)的sql進(jìn)行
    的頭像 發(fā)表于 07-09 17:26 ?373次閱讀
    使用<b class='flag-5'>mybatis</b>切片<b class='flag-5'>實(shí)現(xiàn)</b><b class='flag-5'>數(shù)據(jù)</b><b class='flag-5'>權(quán)限</b>控制