欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

mybatis 實現(xiàn) SQL 查詢攔截修改詳解

 更新時間:2019年07月16日 10:33:56   作者:coderstory  
這篇文章主要介紹了mybatis 實現(xiàn) SQL 查詢攔截修改詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下

前言

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

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

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

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

接口中包含了三個方法定義

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

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

當執(zhí)行代理對象的接口方法時,會調(diào)用Plugin的invoke方法,它會把要執(zhí)行的對象,方法和參數(shù)打包成Invocation對象傳給攔截器的intercept方法。Invocation定義了一個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 方法顧名思義,用于設置屬性的。bean的屬性初始化方法有很多,這是其中的一種。

mybatis提供了@Intercepts注解用于聲明當前類是攔截器,其值為@Signature數(shù)組,表明要攔截的接口、方法以及對應的參數(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 {
.....

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

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

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

  <!-- 配置mybatis -->
  <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
    <property name="dataSource" ref="dataSource"/>
    <property name="configLocation" value="classpath:mybatis/mybatis-config.xml"/>
    <!-- mapper掃描 -->
    <property name="mapperLocations" value="classpath:mybatis/*/*.xml"/>
    <property name="plugins">
      <array>
        <!-- 注冊自己的攔截器 -->
        <bean id="paginationInterceptor" class="xxx.xxx.TenantInterceptor">
        </bean>
      </array>
    </property>
  </bean>

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關文章

  • MySQL5.7.16綠色版安裝教程詳解

    MySQL5.7.16綠色版安裝教程詳解

    這篇文章主要介紹了MySQL5.7.16綠色版安裝教程詳解的相關資料,本文給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下
    2016-10-10
  • MySQL筆記之視圖的使用詳解

    MySQL筆記之視圖的使用詳解

    使用視圖的大部分情況是為了保障數(shù)據(jù)安全性,提高查詢效率
    2013-05-05
  • mysql免安裝沒有任何root權限

    mysql免安裝沒有任何root權限

    這篇文章主要介紹了mysql免安裝沒有任何root權限的安裝與配置教程,本文圖文并茂給大家介紹的非常詳細,需要的朋友參考下
    2017-01-01
  • 解讀MySQL為什么不推薦使用外鍵

    解讀MySQL為什么不推薦使用外鍵

    這篇文章主要介紹了解讀MySQL為什么不推薦使用外鍵問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-04-04
  • 淺談Mysql中類似于nvl()函數(shù)的ifnull()函數(shù)

    淺談Mysql中類似于nvl()函數(shù)的ifnull()函數(shù)

    下面小編就為大家?guī)硪黄獪\談Mysql中類似于nvl()函數(shù)的ifnull()函數(shù)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-02-02
  • 安全地關閉MySQL服務的教程

    安全地關閉MySQL服務的教程

    這篇文章主要介紹了安全地關閉MySQL服務的教程,包括使用kill指令時的一些注意點,需要的朋友可以參考下
    2015-11-11
  • Mysql數(shù)據(jù)庫鎖定機制詳細介紹

    Mysql數(shù)據(jù)庫鎖定機制詳細介紹

    這篇文章主要介紹了Mysql數(shù)據(jù)庫鎖定機制詳細介紹,本文用大量內(nèi)容講解了Mysql中的鎖定機制,例如MySQL鎖定機制簡介、合理利用鎖機制優(yōu)化MySQL等內(nèi)容,需要的朋友可以參考下
    2014-12-12
  • MySQL中l(wèi)imit語法及用法小結

    MySQL中l(wèi)imit語法及用法小結

    LIMIT 是 MySQL 中的一個特殊關鍵字,用于指定查詢結果從哪條記錄開始顯示,一共顯示多少條記錄,本文重點介紹MySQL中l(wèi)imit語法及用法小結,感興趣的朋友一起看看吧
    2023-10-10
  • Spark SQL常見4種數(shù)據(jù)源詳解

    Spark SQL常見4種數(shù)據(jù)源詳解

    這篇文章主要介紹了Spark SQL常見4種數(shù)據(jù)源(詳細),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,,需要的朋友可以參考下
    2019-06-06
  • win2003 mysql單向同步配置步驟[已測]

    win2003 mysql單向同步配置步驟[已測]

    為了服務器安全特將mysql同步了下,由于時間長了經(jīng)常忘記,特以此整理下了,方便經(jīng)常支持腳本之家的朋友。
    2011-06-06

最新評論