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

詳解Java的MyBatis框架中的緩存與緩存的使用改進

 更新時間:2016年06月01日 18:42:10   作者:亦山  
很多人在使用MyBatis的緩存后經(jīng)常會遇到MySQL分頁查詢的顯示問題,針對于此,這里我們就來詳解Java的MyBatis框架中的緩存與緩存的使用改進,首先來回顧一下MyBatis的緩存機制與執(zhí)行:

一級緩存與二級緩存
MyBatis將數(shù)據(jù)緩存設計成兩級結構,分為一級緩存、二級緩存:
一級緩存是Session會話級別的緩存,位于表示一次數(shù)據(jù)庫會話的SqlSession對象之中,又被稱之為本地緩存。一級緩存是MyBatis內部實現(xiàn)的一個特性,用戶不能配置,默認情況下自動支持的緩存,用戶沒有定制它的權利(不過這也不是絕對的,可以通過開發(fā)插件對它進行修改);
二級緩存是Application應用級別的緩存,它的是生命周期很長,跟Application的聲明周期一樣,也就是說它的作用范圍是整個Application應用。
MyBatis中一級緩存和二級緩存的組織如下圖所示:

201661182830144.jpg (824×776)

一級緩存的工作機制:
一級緩存是Session會話級別的,一般而言,一個SqlSession對象會使用一個Executor對象來完成會話操作,Executor對象會維護一個Cache緩存,以提高查詢性能。
二級緩存的工作機制:
如上所言,一個SqlSession對象會使用一個Executor對象來完成會話操作,MyBatis的二級緩存機制的關鍵就是對這個Executor對象做文章。如果用戶配置了"cacheEnabled=true",那么MyBatis在為SqlSession對象創(chuàng)建Executor對象時,會對Executor對象加上一個裝飾者:CachingExecutor,這時SqlSession使用CachingExecutor對象來完成操作請求。CachingExecutor對于查詢請求,會先判斷該查詢請求在Application級別的二級緩存中是否有緩存結果,如果有查詢結果,則直接返回緩存結果;如果緩存中沒有,再交給真正的Executor對象來完成查詢操作,之后CachingExecutor會將真正Executor返回的查詢結果放置到緩存中,然后在返回給用戶。
MyBatis的二級緩存設計得比較靈活,你可以使用MyBatis自己定義的二級緩存實現(xiàn);你也可以通過實現(xiàn)org.apache.ibatis.cache.Cache接口自定義緩存;也可以使用第三方內存緩存庫,如Memcached等。

201661182937460.jpg (755×407)

201661182952328.jpg (544×421)

緩存的改造
問題:
最容易出現(xiàn)的問題是開啟cache后,分頁查詢時無論查詢哪一頁都返回第一頁的數(shù)據(jù)。另外,使用sql自動生成插件生成get方法的sql時,傳入的參數(shù)不起作用,無論傳入的參數(shù)是多少,都返回第一個參數(shù)的查詢結果。

為什么出現(xiàn)這些問題:
在之前講解Mybatis的執(zhí)行流程的時候提到,在開啟cache的前提下,Mybatis的executor會先從緩存里讀取數(shù)據(jù),讀取不到才去數(shù)據(jù)庫查詢。問題就出在這里,sql自動生成插件和分頁插件執(zhí)行的時機是在statementhandler里,而statementhandler是在executor之后執(zhí)行的,無論sql自動生成插件和分頁插件都是通過改寫sql來實現(xiàn)的,executor在生成讀取cache的key(key由sql以及對應的參數(shù)值構成)時使用都是原始的sql,這樣當然就出問題了。

解決問題:
找到問題的原因后,解決起來就方便了。只要通過攔截器改寫executor里生成key的方法,在生成可以時使用自動生成的sql(對應sql自動生成插件)或加入分頁信息(對應分頁插件)就可以了。

攔截器簽名:

@Intercepts({@Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class})}) 
public class CacheInterceptor implements Interceptor { 
... 
} 

從簽名里可以看出,要攔截的目標類型是Executor(注意:type只能配置成接口類型),攔截的方法是名稱為query的方法。

intercept的實現(xiàn):

public Object intercept(Invocation invocation) throws Throwable { 
    Executor executorProxy = (Executor) invocation.getTarget(); 
    MetaObject metaExecutor = MetaObject.forObject(executorProxy, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY); 
    // 分離代理對象鏈 
    while (metaExecutor.hasGetter("h")) { 
      Object object = metaExecutor.getValue("h"); 
      metaExecutor = MetaObject.forObject(object, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY); 
    } 
    // 分離最后一個代理對象的目標類 
    while (metaExecutor.hasGetter("target")) { 
      Object object = metaExecutor.getValue("target"); 
      metaExecutor = MetaObject.forObject(object, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY); 
    } 
    Object[] args = invocation.getArgs(); 
    return this.query(metaExecutor, args); 
  } 
 
  public <E> List<E> query(MetaObject metaExecutor, Object[] args) throws SQLException { 
    MappedStatement ms = (MappedStatement) args[0]; 
    Object parameterObject = args[1]; 
    RowBounds rowBounds = (RowBounds) args[2]; 
    ResultHandler resultHandler = (ResultHandler) args[3]; 
    BoundSql boundSql = ms.getBoundSql(parameterObject); 
    // 改寫key的生成 
    CacheKey cacheKey = createCacheKey(ms, parameterObject, rowBounds, boundSql); 
    Executor executor = (Executor) metaExecutor.getOriginalObject(); 
    return executor.query(ms, parameterObject, rowBounds, resultHandler, cacheKey, boundSql); 
  } 
 
  private CacheKey createCacheKey(MappedStatement ms, Object parameterObject, RowBounds rowBounds, BoundSql boundSql) { 
    Configuration configuration = ms.getConfiguration(); 
    pageSqlId = configuration.getVariables().getProperty("pageSqlId"); 
    if (null == pageSqlId || "".equals(pageSqlId)) { 
      logger.warn("Property pageSqlId is not setted,use default '.*Page$' "); 
      pageSqlId = defaultPageSqlId; 
    } 
    CacheKey cacheKey = new CacheKey(); 
    cacheKey.update(ms.getId()); 
    cacheKey.update(rowBounds.getOffset()); 
    cacheKey.update(rowBounds.getLimit()); 
    List<ParameterMapping> parameterMappings = boundSql.getParameterMappings(); 
    // 解決自動生成SQL,SQL語句為空導致key生成錯誤的bug 
    if (null == boundSql.getSql() || "".equals(boundSql.getSql())) { 
      String id = ms.getId(); 
      id = id.substring(id.lastIndexOf(".") + 1); 
      String newSql = null; 
      try { 
        if ("select".equals(id)) { 
          newSql = SqlBuilder.buildSelectSql(parameterObject); 
        } 
        SqlSource sqlSource = buildSqlSource(configuration, newSql, parameterObject.getClass()); 
        parameterMappings = sqlSource.getBoundSql(parameterObject).getParameterMappings(); 
        cacheKey.update(newSql); 
      } catch (Exception e) { 
        logger.error("Update cacheKey error.", e); 
      } 
    } else { 
      cacheKey.update(boundSql.getSql()); 
    } 
 
    MetaObject metaObject = MetaObject.forObject(parameterObject, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY); 
 
    if (parameterMappings.size() > 0 && parameterObject != null) { 
      TypeHandlerRegistry typeHandlerRegistry = ms.getConfiguration().getTypeHandlerRegistry(); 
      if (typeHandlerRegistry.hasTypeHandler(parameterObject.getClass())) { 
        cacheKey.update(parameterObject); 
      } else { 
        for (ParameterMapping parameterMapping : parameterMappings) { 
          String propertyName = parameterMapping.getProperty(); 
          if (metaObject.hasGetter(propertyName)) { 
            cacheKey.update(metaObject.getValue(propertyName)); 
          } else if (boundSql.hasAdditionalParameter(propertyName)) { 
            cacheKey.update(boundSql.getAdditionalParameter(propertyName)); 
          } 
        } 
      } 
    } 
    // 當需要分頁查詢時,將page參數(shù)里的當前頁和每頁數(shù)加到cachekey里 
    if (ms.getId().matches(pageSqlId) && metaObject.hasGetter("page")) { 
      PageParameter page = (PageParameter) metaObject.getValue("page"); 
      if (null != page) { 
        cacheKey.update(page.getCurrentPage()); 
        cacheKey.update(page.getPageSize()); 
      } 
    } 
    return cacheKey; 
}  

plugin的實現(xiàn):

public Object plugin(Object target) { 
  // 當目標類是CachingExecutor類型時,才包裝目標類,否者直接返回目標本身,減少目標被代理的 
  // 次數(shù) 
  if (target instanceof CachingExecutor) { 
    return Plugin.wrap(target, this); 
  } else { 
    return target; 
  } 
} 

相關文章

  • Java中Integer類型值相等判斷方法

    Java中Integer類型值相等判斷方法

    這篇文章主要給大家介紹了關于Java中Integer類型值相等判斷的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-02-02
  • java實現(xiàn)微信點餐申請微信退款

    java實現(xiàn)微信點餐申請微信退款

    這篇文章主要為大家詳細介紹了java實現(xiàn)微信點餐申請微信退款,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-09-09
  • 使用SpringBoot進行身份驗證和授權的示例詳解

    使用SpringBoot進行身份驗證和授權的示例詳解

    在廣闊的 Web 開發(fā)世界中,身份驗證是每個數(shù)字領域的守護者,在本教程中,我們將了解如何以本機方式保護、驗證和授權 Spring-Boot 應用程序的用戶,并遵循框架的良好實踐,希望對大家有所幫助
    2023-11-11
  • spark之Standalone模式部署配置詳解

    spark之Standalone模式部署配置詳解

    這篇文章主要介紹了spark之Standalone模式部署配置詳解,小編覺得挺不錯的,這里分享給大家,供各位參考。
    2017-10-10
  • SSH 框架簡介

    SSH 框架簡介

    SSH是 struts+spring+hibernate的一個集成框架,是目前較流行的一種web應用程序開源框架。本文給大家詳細看一下組成SSH的這三個框架
    2017-09-09
  • Mybatis的@select和@SelectProvider注解方式動態(tài)SQL語句解讀

    Mybatis的@select和@SelectProvider注解方式動態(tài)SQL語句解讀

    這篇文章主要介紹了Mybatis的@select和@SelectProvider注解方式動態(tài)SQL語句,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-12-12
  • spring整合JMS實現(xiàn)同步收發(fā)消息(基于ActiveMQ的實現(xiàn))

    spring整合JMS實現(xiàn)同步收發(fā)消息(基于ActiveMQ的實現(xiàn))

    本篇文章主要介紹了spring整合JMS實現(xiàn)同步收發(fā)消息(基于ActiveMQ的實現(xiàn)),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-10-10
  • Java將文件按照指定格式切分成多個文件

    Java將文件按照指定格式切分成多個文件

    這篇文章主要為大家詳細介紹了Java如何將文件按照指定格式切分成多個文件,文中的示例代碼簡潔易懂,有需要的小伙伴可以參考一下
    2025-03-03
  • 使用maven插件對java工程進行打包過程解析

    使用maven插件對java工程進行打包過程解析

    這篇文章主要介紹了使用maven插件對java工程進行打包過程解析,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2019-08-08
  • Java編譯時類型與運行時類型

    Java編譯時類型與運行時類型

    這篇文章主要介紹了Java編譯時類型與運行時類型,文章以父類BaseClass和子類SubClass為例展開對主題的探討,具有一的?參考價值,需要的小伙伴可以參考一下
    2022-03-03

最新評論