久久精品人人爽,华人av在线,亚洲性视频网站,欧美专区一二三

mybatis如何實(shí)現(xiàn)SQL查詢攔截修改

167次閱讀
沒有評論

共計(jì) 3865 個(gè)字符,預(yù)計(jì)需要花費(fèi) 10 分鐘才能閱讀完成。

自動寫代碼機(jī)器人,免費(fèi)開通

這篇文章將為大家詳細(xì)講解有關(guān) mybatis 如何實(shí)現(xiàn) SQL 查詢攔截修改,丸趣 TV 小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考,希望大家閱讀完這篇文章后可以有所收獲。

前言

截器的一個(gè)作用就是我們可以攔截某些方法的調(diào)用,我們可以選擇在這些被攔截的方法執(zhí)行前后加上某些邏輯,也可以在執(zhí)行這些被攔截的方法時(shí)執(zhí)行自己的邏輯而不再執(zhí)行被攔截的方法。

Mybatis 攔截器設(shè)計(jì)的一個(gè)初衷就是為了供用戶在某些時(shí)候可以實(shí)現(xiàn)自己的邏輯而不必去動 Mybatis 固有的邏輯。比如我想針對所有的 SQL 執(zhí)行某個(gè)固定的操作,針對 SQL 查詢執(zhí)行安全檢查,或者記錄相關(guān) SQL 查詢?nèi)罩镜鹊取?/p>

Mybatis 為我們提供了一個(gè) Interceptor 接口,可以實(shí)現(xiàn)自定義的攔截器。

 public interface Interceptor { Object intercept(Invocation invocation) throws Throwable;
 Object plugin(Object target);
 void setProperties(Properties properties);
}

接口中包含了三個(gè)方法定義

intercept 方法為具體的攔截對象的處理方法,傳入的 Invocation 包含了攔截目標(biāo)類的實(shí)力,攔截的方法和方法的入?yún)?shù)組。使用 Invocation 的 procced 執(zhí)行原函數(shù)。

plugin 中執(zhí)行判斷是否要進(jìn)行攔截進(jìn),如果不需要攔截,直接返回 target,如果需要攔截則調(diào)用 Plugin 類中的 wrap 靜態(tài)方法,如果當(dāng)前攔截器實(shí)現(xiàn)了任意接口,則返回一個(gè)代理對象,否則直接返回(回憶代理模式的設(shè)計(jì))。代理對象實(shí)際是一個(gè) Plugin 類實(shí)例,它實(shí)現(xiàn)了 InvocationHandler 接口,InvocationHandler 接口僅包含 invoke 方法用于回調(diào)方法。

當(dāng)執(zhí)行代理對象的接口方法時(shí),會調(diào)用 Plugin 的 invoke 方法,它會把要執(zhí)行的對象,方法和參數(shù)打包成 Invocation 對象傳給攔截器的 intercept 方法。Invocation 定義了一個(gè) procced 方法,用于執(zhí)行被攔截的原方法。

Plugin 類定義

public class Plugin implements InvocationHandler {
 
 private Object target;
 private Interceptor interceptor;
 private Map, Set  signatureMap;
 
 private Plugin(Object target, Interceptor interceptor, Map, Set  signatureMap) {
 this.target = target;
 this.interceptor = interceptor;
 this.signatureMap = signatureMap;
 }
 
 public static Object wrap(Object target, Interceptor interceptor) { Map, Set  signatureMap = getSignatureMap(interceptor);
 Class type = target.getClass();
 Class[] interfaces = getAllInterfaces(type, signatureMap);
 if (interfaces.length   0) {
 return Proxy.newProxyInstance( type.getClassLoader(),
 interfaces,
 new Plugin(target, interceptor, signatureMap));
 }
 return target;
 }
 
 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
 try { Set methods = signatureMap.get(method.getDeclaringClass());
 if (methods != null   methods.contains(method)) { return interceptor.intercept(new Invocation(target, method, args));
 }
 return method.invoke(target, args);
 } catch (Exception e) { throw ExceptionUtil.unwrapThrowable(e);
 }
 }
 
 private static Map, Set  getSignatureMap(Interceptor interceptor) { Intercepts interceptsAnnotation = interceptor.getClass().getAnnotation(Intercepts.class);
 if (interceptsAnnotation == null) { // issue #251
 throw new PluginException(No @Intercepts annotation was found in interceptor   + interceptor.getClass().getName()); 
 }
 Signature[] sigs = interceptsAnnotation.value();
 Map, Set  signatureMap = new HashMap, Set 
 for (Signature sig : sigs) { Set methods = signatureMap.get(sig.type());
 if (methods == null) { methods = new HashSet();
 signatureMap.put(sig.type(), methods);
 }
 try { Method method = sig.type().getMethod(sig.method(), sig.args());
 methods.add(method);
 } catch (NoSuchMethodException e) { throw new PluginException( Could not find method on   + sig.type() +   named   + sig.method() +  . Cause:   + e, e);
 }
 }
 return signatureMap;
 }
 
 private static Class[] getAllInterfaces(Class type, Map, Set  signatureMap) {
 Set  interfaces = new HashSet 
 while (type != null) { for (Class c : type.getInterfaces()) { if (signatureMap.containsKey(c)) { interfaces.add(c);
 }
 }
 type = type.getSuperclass();
 }
 return interfaces.toArray(new Class[interfaces.size()]);
 }
 
}

setProperties 方法顧名思義,用于設(shè)置屬性的。bean 的屬性初始化方法有很多,這是其中的一種。

mybatis 提供了 @Intercepts 注解用于聲明當(dāng)前類是攔截器,其值為 @Signature 數(shù)組,表明要攔截的接口、方法以及對應(yīng)的參數(shù)類型

@Intercepts({@Signature(method =  prepare , type = StatementHandler.class, args = {Connection.class}),
 @Signature(method =  query , type = StatementHandler.class, args = {java.sql.Statement.class, ResultHandler.class})})
public class TenantInterceptor implements Interceptor {.....

例如上面的類聲明,第一個(gè) Signature 標(biāo)注攔截了 StatementHandler 類下的入?yún)⑹且粋€(gè) Connection 的名為 prepare 的方法。

第二個(gè) Signature 標(biāo)注攔截 StatementHandler 類中包含 2 個(gè)入?yún)ⅲǚ謩e為 Statement 和 ResultHandler 類型)的名為 query 的方法。

最后,聲明的 Interceptor 需要注冊到 mybatis 的 plug 中才能生效。

  !--  配置 mybatis --
  bean id= sqlSessionFactory   >

關(guān)于“mybatis 如何實(shí)現(xiàn) SQL 查詢攔截修改”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,使各位可以學(xué)到更多知識,如果覺得文章不錯(cuò),請把它分享出去讓更多的人看到。

向 AI 問一下細(xì)節(jié)

正文完
 
丸趣
版權(quán)聲明:本站原創(chuàng)文章,由 丸趣 2023-12-04發(fā)表,共計(jì)3865字。
轉(zhuǎn)載說明:除特殊說明外本站除技術(shù)相關(guān)以外文章皆由網(wǎng)絡(luò)搜集發(fā)布,轉(zhuǎn)載請注明出處。
評論(沒有評論)
主站蜘蛛池模板: 武定县| 永靖县| 余干县| 旺苍县| 宝丰县| 琼结县| 锡林浩特市| 黑水县| 萝北县| 吉木萨尔县| 开江县| 博爱县| 白城市| 河津市| 新津县| 通城县| 科技| 即墨市| 雅江县| 阿荣旗| 双牌县| 定州市| 河源市| 博乐市| 江孜县| 达孜县| 龙陵县| 鄂托克前旗| 新竹县| 上饶市| 宜良县| 河间市| 新建县| 儋州市| 乌什县| 星座| 湘西| 遂宁市| 许昌市| 淮安市| 安徽省|